|  | @@ -1,13 +1,15 @@
 | 
											
												
													
														|  |  import datetime
 |  |  import datetime
 | 
											
												
													
														|  | -import time
 |  | 
 | 
											
												
													
														|  |  import itertools
 |  |  import itertools
 | 
											
												
													
														|  |  import logging
 |  |  import logging
 | 
											
												
													
														|  | 
 |  | +import multiprocessing as mp
 | 
											
												
													
														|  |  import os
 |  |  import os
 | 
											
												
													
														|  | 
 |  | +import queue
 | 
											
												
													
														|  |  import random
 |  |  import random
 | 
											
												
													
														|  |  import shutil
 |  |  import shutil
 | 
											
												
													
														|  |  import string
 |  |  import string
 | 
											
												
													
														|  |  import subprocess as sp
 |  |  import subprocess as sp
 | 
											
												
													
														|  |  import threading
 |  |  import threading
 | 
											
												
													
														|  | 
 |  | +import time
 | 
											
												
													
														|  |  from collections import defaultdict
 |  |  from collections import defaultdict
 | 
											
												
													
														|  |  from pathlib import Path
 |  |  from pathlib import Path
 | 
											
												
													
														|  |  
 |  |  
 | 
											
										
											
												
													
														|  | @@ -40,22 +42,28 @@ def remove_empty_directories(directory):
 | 
											
												
													
														|  |  
 |  |  
 | 
											
												
													
														|  |  
 |  |  
 | 
											
												
													
														|  |  class RecordingMaintainer(threading.Thread):
 |  |  class RecordingMaintainer(threading.Thread):
 | 
											
												
													
														|  | -    def __init__(self, config: FrigateConfig, stop_event):
 |  | 
 | 
											
												
													
														|  | 
 |  | +    def __init__(
 | 
											
												
													
														|  | 
 |  | +        self, config: FrigateConfig, recordings_info_queue: mp.Queue, stop_event
 | 
											
												
													
														|  | 
 |  | +    ):
 | 
											
												
													
														|  |          threading.Thread.__init__(self)
 |  |          threading.Thread.__init__(self)
 | 
											
												
													
														|  |          self.name = "recording_maint"
 |  |          self.name = "recording_maint"
 | 
											
												
													
														|  |          self.config = config
 |  |          self.config = config
 | 
											
												
													
														|  | 
 |  | +        self.recordings_info_queue = recordings_info_queue
 | 
											
												
													
														|  |          self.stop_event = stop_event
 |  |          self.stop_event = stop_event
 | 
											
												
													
														|  |          self.first_pass = True
 |  |          self.first_pass = True
 | 
											
												
													
														|  | 
 |  | +        self.recordings_info = defaultdict(list)
 | 
											
												
													
														|  |          self.end_time_cache = {}
 |  |          self.end_time_cache = {}
 | 
											
												
													
														|  |  
 |  |  
 | 
											
												
													
														|  |      def move_files(self):
 |  |      def move_files(self):
 | 
											
												
													
														|  | -        cache_files = [
 |  | 
 | 
											
												
													
														|  | -            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_")
 |  | 
 | 
											
												
													
														|  | -        ]
 |  | 
 | 
											
												
													
														|  | 
 |  | +        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_")
 | 
											
												
													
														|  | 
 |  | +            ]
 | 
											
												
													
														|  | 
 |  | +        )
 | 
											
												
													
														|  |  
 |  |  
 | 
											
												
													
														|  |          files_in_use = []
 |  |          files_in_use = []
 | 
											
												
													
														|  |          for process in psutil.process_iter():
 |  |          for process in psutil.process_iter():
 | 
											
										
											
												
													
														|  | @@ -93,16 +101,22 @@ class RecordingMaintainer(threading.Thread):
 | 
											
												
													
														|  |          keep_count = 5
 |  |          keep_count = 5
 | 
											
												
													
														|  |          for camera in grouped_recordings.keys():
 |  |          for camera in grouped_recordings.keys():
 | 
											
												
													
														|  |              if len(grouped_recordings[camera]) > keep_count:
 |  |              if len(grouped_recordings[camera]) > keep_count:
 | 
											
												
													
														|  | -                sorted_recordings = sorted(
 |  | 
 | 
											
												
													
														|  | -                    grouped_recordings[camera], key=lambda i: i["start_time"]
 |  | 
 | 
											
												
													
														|  | -                )
 |  | 
 | 
											
												
													
														|  | -                to_remove = sorted_recordings[:-keep_count]
 |  | 
 | 
											
												
													
														|  | 
 |  | +                to_remove = grouped_recordings[camera][:-keep_count]
 | 
											
												
													
														|  |                  for f in to_remove:
 |  |                  for f in to_remove:
 | 
											
												
													
														|  |                      Path(f["cache_path"]).unlink(missing_ok=True)
 |  |                      Path(f["cache_path"]).unlink(missing_ok=True)
 | 
											
												
													
														|  |                      self.end_time_cache.pop(f["cache_path"], None)
 |  |                      self.end_time_cache.pop(f["cache_path"], None)
 | 
											
												
													
														|  | -                grouped_recordings[camera] = sorted_recordings[-keep_count:]
 |  | 
 | 
											
												
													
														|  | 
 |  | +                grouped_recordings[camera] = grouped_recordings[camera][-keep_count:]
 | 
											
												
													
														|  |  
 |  |  
 | 
											
												
													
														|  |          for camera, recordings in grouped_recordings.items():
 |  |          for camera, recordings in grouped_recordings.items():
 | 
											
												
													
														|  | 
 |  | +
 | 
											
												
													
														|  | 
 |  | +            # 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)
 | 
											
												
													
														|  | 
 |  | +
 | 
											
												
													
														|  |              # get all events with the end time after the start of the oldest cache file
 |  |              # get all events with the end time after the start of the oldest cache file
 | 
											
												
													
														|  |              # or with end_time None
 |  |              # or with end_time None
 | 
											
												
													
														|  |              events: Event = (
 |  |              events: Event = (
 | 
											
										
											
												
													
														|  | @@ -167,6 +181,8 @@ class RecordingMaintainer(threading.Thread):
 | 
											
												
													
														|  |                          # and remove this segment
 |  |                          # and remove this segment
 | 
											
												
													
														|  |                          if event.start_time > end_time.timestamp():
 |  |                          if event.start_time > end_time.timestamp():
 | 
											
												
													
														|  |                              overlaps = False
 |  |                              overlaps = False
 | 
											
												
													
														|  | 
 |  | +                            Path(cache_path).unlink(missing_ok=True)
 | 
											
												
													
														|  | 
 |  | +                            self.end_time_cache.pop(cache_path, None)
 | 
											
												
													
														|  |                              break
 |  |                              break
 | 
											
												
													
														|  |  
 |  |  
 | 
											
												
													
														|  |                          # if the event is in progress or ends after the recording starts, keep it
 |  |                          # if the event is in progress or ends after the recording starts, keep it
 | 
											
										
											
												
													
														|  | @@ -235,6 +251,30 @@ class RecordingMaintainer(threading.Thread):
 | 
											
												
													
														|  |          wait_time = 5
 |  |          wait_time = 5
 | 
											
												
													
														|  |          while not self.stop_event.wait(wait_time):
 |  |          while not self.stop_event.wait(wait_time):
 | 
											
												
													
														|  |              run_start = datetime.datetime.now().timestamp()
 |  |              run_start = datetime.datetime.now().timestamp()
 | 
											
												
													
														|  | 
 |  | +
 | 
											
												
													
														|  | 
 |  | +            # empty the recordings info queue
 | 
											
												
													
														|  | 
 |  | +            while True:
 | 
											
												
													
														|  | 
 |  | +                try:
 | 
											
												
													
														|  | 
 |  | +                    (
 | 
											
												
													
														|  | 
 |  | +                        camera,
 | 
											
												
													
														|  | 
 |  | +                        frame_time,
 | 
											
												
													
														|  | 
 |  | +                        current_tracked_objects,
 | 
											
												
													
														|  | 
 |  | +                        motion_boxes,
 | 
											
												
													
														|  | 
 |  | +                        regions,
 | 
											
												
													
														|  | 
 |  | +                    ) = self.recordings_info_queue.get(False)
 | 
											
												
													
														|  | 
 |  | +
 | 
											
												
													
														|  | 
 |  | +                    if self.config.cameras[camera].record.enabled:
 | 
											
												
													
														|  | 
 |  | +                        self.recordings_info[camera].append(
 | 
											
												
													
														|  | 
 |  | +                            (
 | 
											
												
													
														|  | 
 |  | +                                frame_time,
 | 
											
												
													
														|  | 
 |  | +                                current_tracked_objects,
 | 
											
												
													
														|  | 
 |  | +                                motion_boxes,
 | 
											
												
													
														|  | 
 |  | +                                regions,
 | 
											
												
													
														|  | 
 |  | +                            )
 | 
											
												
													
														|  | 
 |  | +                        )
 | 
											
												
													
														|  | 
 |  | +                except queue.Empty:
 | 
											
												
													
														|  | 
 |  | +                    break
 | 
											
												
													
														|  | 
 |  | +
 | 
											
												
													
														|  |              try:
 |  |              try:
 | 
											
												
													
														|  |                  self.move_files()
 |  |                  self.move_files()
 | 
											
												
													
														|  |              except Exception as e:
 |  |              except Exception as e:
 |