app.py 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386
  1. import json
  2. import logging
  3. import multiprocessing as mp
  4. import os
  5. import signal
  6. import sys
  7. import threading
  8. from logging.handlers import QueueHandler
  9. from typing import Dict, List
  10. import yaml
  11. from peewee_migrate import Router
  12. from playhouse.sqlite_ext import SqliteExtDatabase
  13. from playhouse.sqliteq import SqliteQueueDatabase
  14. from frigate.config import DetectorTypeEnum, FrigateConfig
  15. from frigate.const import CACHE_DIR, CLIPS_DIR, RECORD_DIR
  16. from frigate.edgetpu import EdgeTPUProcess
  17. from frigate.events import EventCleanup, EventProcessor
  18. from frigate.http import create_app
  19. from frigate.log import log_process, root_configurer
  20. from frigate.models import Event, Recordings
  21. from frigate.mqtt import MqttSocketRelay, create_mqtt_client
  22. from frigate.object_processing import TrackedObjectProcessor
  23. from frigate.output import output_frames
  24. from frigate.record import RecordingCleanup, RecordingMaintainer
  25. from frigate.stats import StatsEmitter, stats_init
  26. from frigate.version import VERSION
  27. from frigate.video import capture_camera, track_camera
  28. from frigate.watchdog import FrigateWatchdog
  29. logger = logging.getLogger(__name__)
  30. class FrigateApp:
  31. def __init__(self):
  32. self.stop_event = mp.Event()
  33. self.base_config: FrigateConfig = None
  34. self.config: FrigateConfig = None
  35. self.detection_queue = mp.Queue()
  36. self.detectors: Dict[str, EdgeTPUProcess] = {}
  37. self.detection_out_events: Dict[str, mp.Event] = {}
  38. self.detection_shms: List[mp.shared_memory.SharedMemory] = []
  39. self.log_queue = mp.Queue()
  40. self.camera_metrics = {}
  41. def set_environment_vars(self):
  42. for key, value in self.config.environment_vars.items():
  43. os.environ[key] = value
  44. def ensure_dirs(self):
  45. for d in [RECORD_DIR, CLIPS_DIR, CACHE_DIR]:
  46. if not os.path.exists(d) and not os.path.islink(d):
  47. logger.info(f"Creating directory: {d}")
  48. os.makedirs(d)
  49. else:
  50. logger.debug(f"Skipping directory: {d}")
  51. def init_logger(self):
  52. self.log_process = mp.Process(
  53. target=log_process, args=(self.log_queue,), name="log_process"
  54. )
  55. self.log_process.daemon = True
  56. self.log_process.start()
  57. root_configurer(self.log_queue)
  58. def init_config(self):
  59. config_file = os.environ.get("CONFIG_FILE", "/config/config.yml")
  60. user_config = FrigateConfig.parse_file(config_file)
  61. self.config = user_config.runtime_config
  62. for camera_name in self.config.cameras.keys():
  63. # create camera_metrics
  64. self.camera_metrics[camera_name] = {
  65. "camera_fps": mp.Value("d", 0.0),
  66. "skipped_fps": mp.Value("d", 0.0),
  67. "process_fps": mp.Value("d", 0.0),
  68. "detection_enabled": mp.Value(
  69. "i", self.config.cameras[camera_name].detect.enabled
  70. ),
  71. "detection_fps": mp.Value("d", 0.0),
  72. "detection_frame": mp.Value("d", 0.0),
  73. "read_start": mp.Value("d", 0.0),
  74. "ffmpeg_pid": mp.Value("i", 0),
  75. "frame_queue": mp.Queue(maxsize=2),
  76. }
  77. def check_config(self):
  78. for name, camera in self.config.cameras.items():
  79. assigned_roles = list(
  80. set([r for i in camera.ffmpeg.inputs for r in i.roles])
  81. )
  82. if not camera.record.enabled and "record" in assigned_roles:
  83. logger.warning(
  84. f"Camera {name} has record assigned to an input, but record is not enabled."
  85. )
  86. elif camera.record.enabled and not "record" in assigned_roles:
  87. logger.warning(
  88. f"Camera {name} has record enabled, but record is not assigned to an input."
  89. )
  90. if not camera.rtmp.enabled and "rtmp" in assigned_roles:
  91. logger.warning(
  92. f"Camera {name} has rtmp assigned to an input, but rtmp is not enabled."
  93. )
  94. elif camera.rtmp.enabled and not "rtmp" in assigned_roles:
  95. logger.warning(
  96. f"Camera {name} has rtmp enabled, but rtmp is not assigned to an input."
  97. )
  98. def set_log_levels(self):
  99. logging.getLogger().setLevel(self.config.logger.default.value.upper())
  100. for log, level in self.config.logger.logs.items():
  101. logging.getLogger(log).setLevel(level.value.upper())
  102. if not "werkzeug" in self.config.logger.logs:
  103. logging.getLogger("werkzeug").setLevel("ERROR")
  104. def init_queues(self):
  105. # Queues for clip processing
  106. self.event_queue = mp.Queue()
  107. self.event_processed_queue = mp.Queue()
  108. self.video_output_queue = mp.Queue(maxsize=len(self.config.cameras.keys()) * 2)
  109. # Queue for cameras to push tracked objects to
  110. self.detected_frames_queue = mp.Queue(
  111. maxsize=len(self.config.cameras.keys()) * 2
  112. )
  113. def init_database(self):
  114. # Migrate DB location
  115. old_db_path = os.path.join(CLIPS_DIR, "frigate.db")
  116. if not os.path.isfile(self.config.database.path) and os.path.isfile(
  117. old_db_path
  118. ):
  119. os.rename(old_db_path, self.config.database.path)
  120. # Migrate DB schema
  121. migrate_db = SqliteExtDatabase(self.config.database.path)
  122. # Run migrations
  123. del logging.getLogger("peewee_migrate").handlers[:]
  124. router = Router(migrate_db)
  125. router.run()
  126. migrate_db.close()
  127. self.db = SqliteQueueDatabase(self.config.database.path)
  128. models = [Event, Recordings]
  129. self.db.bind(models)
  130. def init_stats(self):
  131. self.stats_tracking = stats_init(self.camera_metrics, self.detectors)
  132. def init_web_server(self):
  133. self.flask_app = create_app(
  134. self.config,
  135. self.db,
  136. self.stats_tracking,
  137. self.detected_frames_processor,
  138. )
  139. def init_mqtt(self):
  140. self.mqtt_client = create_mqtt_client(self.config, self.camera_metrics)
  141. def start_mqtt_relay(self):
  142. self.mqtt_relay = MqttSocketRelay(
  143. self.mqtt_client, self.config.mqtt.topic_prefix
  144. )
  145. self.mqtt_relay.start()
  146. def start_detectors(self):
  147. model_path = self.config.model.path
  148. model_shape = (self.config.model.height, self.config.model.width)
  149. for name in self.config.cameras.keys():
  150. self.detection_out_events[name] = mp.Event()
  151. try:
  152. shm_in = mp.shared_memory.SharedMemory(
  153. name=name,
  154. create=True,
  155. size=self.config.model.height * self.config.model.width * 3,
  156. )
  157. except FileExistsError:
  158. shm_in = mp.shared_memory.SharedMemory(name=name)
  159. try:
  160. shm_out = mp.shared_memory.SharedMemory(
  161. name=f"out-{name}", create=True, size=20 * 6 * 4
  162. )
  163. except FileExistsError:
  164. shm_out = mp.shared_memory.SharedMemory(name=f"out-{name}")
  165. self.detection_shms.append(shm_in)
  166. self.detection_shms.append(shm_out)
  167. for name, detector in self.config.detectors.items():
  168. if detector.type == DetectorTypeEnum.cpu:
  169. self.detectors[name] = EdgeTPUProcess(
  170. name,
  171. self.detection_queue,
  172. self.detection_out_events,
  173. model_path,
  174. model_shape,
  175. "cpu",
  176. detector.num_threads,
  177. )
  178. if detector.type == DetectorTypeEnum.edgetpu:
  179. self.detectors[name] = EdgeTPUProcess(
  180. name,
  181. self.detection_queue,
  182. self.detection_out_events,
  183. model_path,
  184. model_shape,
  185. detector.device,
  186. detector.num_threads,
  187. )
  188. def start_detected_frames_processor(self):
  189. self.detected_frames_processor = TrackedObjectProcessor(
  190. self.config,
  191. self.mqtt_client,
  192. self.config.mqtt.topic_prefix,
  193. self.detected_frames_queue,
  194. self.event_queue,
  195. self.event_processed_queue,
  196. self.video_output_queue,
  197. self.stop_event,
  198. )
  199. self.detected_frames_processor.start()
  200. def start_video_output_processor(self):
  201. output_processor = mp.Process(
  202. target=output_frames,
  203. name=f"output_processor",
  204. args=(
  205. self.config,
  206. self.video_output_queue,
  207. ),
  208. )
  209. output_processor.daemon = True
  210. self.output_processor = output_processor
  211. output_processor.start()
  212. logger.info(f"Output process started: {output_processor.pid}")
  213. def start_camera_processors(self):
  214. model_shape = (self.config.model.height, self.config.model.width)
  215. for name, config in self.config.cameras.items():
  216. camera_process = mp.Process(
  217. target=track_camera,
  218. name=f"camera_processor:{name}",
  219. args=(
  220. name,
  221. config,
  222. model_shape,
  223. self.config.model.merged_labelmap,
  224. self.detection_queue,
  225. self.detection_out_events[name],
  226. self.detected_frames_queue,
  227. self.camera_metrics[name],
  228. ),
  229. )
  230. camera_process.daemon = True
  231. self.camera_metrics[name]["process"] = camera_process
  232. camera_process.start()
  233. logger.info(f"Camera processor started for {name}: {camera_process.pid}")
  234. def start_camera_capture_processes(self):
  235. for name, config in self.config.cameras.items():
  236. capture_process = mp.Process(
  237. target=capture_camera,
  238. name=f"camera_capture:{name}",
  239. args=(name, config, self.camera_metrics[name]),
  240. )
  241. capture_process.daemon = True
  242. self.camera_metrics[name]["capture_process"] = capture_process
  243. capture_process.start()
  244. logger.info(f"Capture process started for {name}: {capture_process.pid}")
  245. def start_event_processor(self):
  246. self.event_processor = EventProcessor(
  247. self.config,
  248. self.camera_metrics,
  249. self.event_queue,
  250. self.event_processed_queue,
  251. self.stop_event,
  252. )
  253. self.event_processor.start()
  254. def start_event_cleanup(self):
  255. self.event_cleanup = EventCleanup(self.config, self.stop_event)
  256. self.event_cleanup.start()
  257. def start_recording_maintainer(self):
  258. self.recording_maintainer = RecordingMaintainer(self.config, self.stop_event)
  259. self.recording_maintainer.start()
  260. def start_recording_cleanup(self):
  261. self.recording_cleanup = RecordingCleanup(self.config, self.stop_event)
  262. self.recording_cleanup.start()
  263. def start_stats_emitter(self):
  264. self.stats_emitter = StatsEmitter(
  265. self.config,
  266. self.stats_tracking,
  267. self.mqtt_client,
  268. self.config.mqtt.topic_prefix,
  269. self.stop_event,
  270. )
  271. self.stats_emitter.start()
  272. def start_watchdog(self):
  273. self.frigate_watchdog = FrigateWatchdog(self.detectors, self.stop_event)
  274. self.frigate_watchdog.start()
  275. def start(self):
  276. self.init_logger()
  277. logger.info(f"Starting Frigate ({VERSION})")
  278. try:
  279. try:
  280. self.init_config()
  281. except Exception as e:
  282. print(f"Error parsing config: {e}")
  283. self.log_process.terminate()
  284. sys.exit(1)
  285. self.set_environment_vars()
  286. self.ensure_dirs()
  287. self.check_config()
  288. self.set_log_levels()
  289. self.init_queues()
  290. self.init_database()
  291. self.init_mqtt()
  292. except Exception as e:
  293. print(e)
  294. self.log_process.terminate()
  295. sys.exit(1)
  296. self.start_detectors()
  297. self.start_video_output_processor()
  298. self.start_detected_frames_processor()
  299. self.start_camera_processors()
  300. self.start_camera_capture_processes()
  301. self.init_stats()
  302. self.init_web_server()
  303. self.start_mqtt_relay()
  304. self.start_event_processor()
  305. self.start_event_cleanup()
  306. self.start_recording_maintainer()
  307. self.start_recording_cleanup()
  308. self.start_stats_emitter()
  309. self.start_watchdog()
  310. # self.zeroconf = broadcast_zeroconf(self.config.mqtt.client_id)
  311. def receiveSignal(signalNumber, frame):
  312. self.stop()
  313. sys.exit()
  314. signal.signal(signal.SIGTERM, receiveSignal)
  315. try:
  316. self.flask_app.run(host="127.0.0.1", port=5001, debug=False)
  317. except KeyboardInterrupt:
  318. pass
  319. self.stop()
  320. def stop(self):
  321. logger.info(f"Stopping...")
  322. self.stop_event.set()
  323. self.mqtt_relay.stop()
  324. self.detected_frames_processor.join()
  325. self.event_processor.join()
  326. self.event_cleanup.join()
  327. self.recording_maintainer.join()
  328. self.recording_cleanup.join()
  329. self.stats_emitter.join()
  330. self.frigate_watchdog.join()
  331. self.db.stop()
  332. for detector in self.detectors.values():
  333. detector.stop()
  334. while len(self.detection_shms) > 0:
  335. shm = self.detection_shms.pop()
  336. shm.close()
  337. shm.unlink()