mirror of
https://github.com/blakeblackshear/frigate.git
synced 2025-09-23 17:52:05 +02:00
* update config for roles and add validator * ensure admin and viewer are never overridden * add class method to user to retrieve all allowed cameras * enforce config roles in auth api endpoints * add camera access api dependency functions * protect review endpoints * protect preview endpoints * rename param name for better fastapi injection matching * remove unneeded * protect export endpoints * protect event endpoints * protect media endpoints * update auth hook for allowed cameras * update default app view * ensure anonymous user always returns all cameras * limit cameras in explore * cameras is already a list * limit cameras in review/history * limit cameras in live view * limit cameras in camera groups * only show face library and classification in sidebar for admin * remove check in delete reviews since admin role is required, no need to check camera access. fixes failing test * pass request with camera access for tests * more async * camera access tests * fix proxy auth tests * allowed cameras for review tests * combine event tests and refactor for camera access * fix post validation for roles * don't limit roles in create user dialog * fix triggers endpoints no need to run require camera access dep since the required role is admin * fix type * create and edit role dialogs * delete role dialog * fix role change dialog * update settings view for roles * i18n changes * minor spacing tweaks * docs * use badges and camera name label component * clarify docs * display all cameras badge for admin and viewer * i18n fix * use validator to prevent reserved and empty roles from being assigned * split users and roles into separate tabs in settings * tweak docs * clarify docs * change icon * don't memoize roles always recalculate on component render
164 lines
5.2 KiB
Python
164 lines
5.2 KiB
Python
from peewee import (
|
|
BlobField,
|
|
BooleanField,
|
|
CharField,
|
|
CompositeKey,
|
|
DateTimeField,
|
|
FloatField,
|
|
ForeignKeyField,
|
|
IntegerField,
|
|
Model,
|
|
TextField,
|
|
)
|
|
from playhouse.sqlite_ext import JSONField
|
|
|
|
|
|
class Event(Model):
|
|
id = CharField(null=False, primary_key=True, max_length=30)
|
|
label = CharField(index=True, max_length=20)
|
|
sub_label = CharField(max_length=100, null=True)
|
|
camera = CharField(index=True, max_length=20)
|
|
start_time = DateTimeField()
|
|
end_time = DateTimeField()
|
|
top_score = (
|
|
FloatField()
|
|
) # TODO remove when columns can be dropped without rebuilding table
|
|
score = (
|
|
FloatField()
|
|
) # TODO remove when columns can be dropped without rebuilding table
|
|
false_positive = BooleanField()
|
|
zones = JSONField()
|
|
thumbnail = TextField()
|
|
has_clip = BooleanField(default=True)
|
|
has_snapshot = BooleanField(default=True)
|
|
region = (
|
|
JSONField()
|
|
) # TODO remove when columns can be dropped without rebuilding table
|
|
box = (
|
|
JSONField()
|
|
) # TODO remove when columns can be dropped without rebuilding table
|
|
area = (
|
|
IntegerField()
|
|
) # TODO remove when columns can be dropped without rebuilding table
|
|
retain_indefinitely = BooleanField(default=False)
|
|
ratio = FloatField(
|
|
default=1.0
|
|
) # TODO remove when columns can be dropped without rebuilding table
|
|
plus_id = CharField(max_length=30)
|
|
model_hash = CharField(max_length=32)
|
|
detector_type = CharField(max_length=32)
|
|
model_type = CharField(max_length=32)
|
|
data = JSONField() # ex: tracked object box, region, etc.
|
|
|
|
|
|
class Timeline(Model):
|
|
timestamp = DateTimeField()
|
|
camera = CharField(index=True, max_length=20)
|
|
source = CharField(index=True, max_length=20) # ex: tracked object, audio, external
|
|
source_id = CharField(index=True, max_length=30)
|
|
class_type = CharField(max_length=50) # ex: entered_zone, audio_heard
|
|
data = JSONField() # ex: tracked object id, region, box, etc.
|
|
|
|
|
|
class Regions(Model):
|
|
camera = CharField(null=False, primary_key=True, max_length=20)
|
|
grid = JSONField() # json blob of grid
|
|
last_update = DateTimeField()
|
|
|
|
|
|
class Recordings(Model):
|
|
id = CharField(null=False, primary_key=True, max_length=30)
|
|
camera = CharField(index=True, max_length=20)
|
|
path = CharField(unique=True)
|
|
start_time = DateTimeField()
|
|
end_time = DateTimeField()
|
|
duration = FloatField()
|
|
motion = IntegerField(null=True)
|
|
objects = IntegerField(null=True)
|
|
dBFS = IntegerField(null=True)
|
|
segment_size = FloatField(default=0) # this should be stored as MB
|
|
regions = IntegerField(null=True)
|
|
|
|
|
|
class Export(Model):
|
|
id = CharField(null=False, primary_key=True, max_length=30)
|
|
camera = CharField(index=True, max_length=20)
|
|
name = CharField(index=True, max_length=100)
|
|
date = DateTimeField()
|
|
video_path = CharField(unique=True)
|
|
thumb_path = CharField(unique=True)
|
|
in_progress = BooleanField()
|
|
|
|
|
|
class ReviewSegment(Model):
|
|
id = CharField(null=False, primary_key=True, max_length=30)
|
|
camera = CharField(index=True, max_length=20)
|
|
start_time = DateTimeField()
|
|
end_time = DateTimeField()
|
|
severity = CharField(max_length=30) # alert, detection
|
|
thumb_path = CharField(unique=True)
|
|
data = JSONField() # additional data about detection like list of labels, zone, areas of significant motion
|
|
|
|
|
|
class UserReviewStatus(Model):
|
|
user_id = CharField(max_length=30)
|
|
review_segment = ForeignKeyField(ReviewSegment, backref="user_reviews")
|
|
has_been_reviewed = BooleanField(default=False)
|
|
|
|
class Meta:
|
|
indexes = ((("user_id", "review_segment"), True),)
|
|
|
|
|
|
class Previews(Model):
|
|
id = CharField(null=False, primary_key=True, max_length=30)
|
|
camera = CharField(index=True, max_length=20)
|
|
path = CharField(unique=True)
|
|
start_time = DateTimeField()
|
|
end_time = DateTimeField()
|
|
duration = FloatField()
|
|
|
|
|
|
# Used for temporary table in record/cleanup.py
|
|
class RecordingsToDelete(Model):
|
|
id = CharField(null=False, primary_key=False, max_length=30)
|
|
|
|
class Meta:
|
|
temporary = True
|
|
|
|
|
|
class User(Model):
|
|
username = CharField(null=False, primary_key=True, max_length=30)
|
|
role = CharField(
|
|
max_length=20,
|
|
default="admin",
|
|
)
|
|
password_hash = CharField(null=False, max_length=120)
|
|
notification_tokens = JSONField()
|
|
|
|
@classmethod
|
|
def get_allowed_cameras(
|
|
cls, role: str, roles_dict: dict[str, list[str]], all_camera_names: set[str]
|
|
) -> list[str]:
|
|
if role not in roles_dict:
|
|
return [] # Invalid role grants no access
|
|
allowed = roles_dict[role]
|
|
if not allowed: # Empty list means all cameras
|
|
return list(all_camera_names)
|
|
|
|
return [cam for cam in allowed if cam in all_camera_names]
|
|
|
|
|
|
class Trigger(Model):
|
|
camera = CharField(max_length=20)
|
|
name = CharField()
|
|
type = CharField(max_length=10)
|
|
data = TextField()
|
|
threshold = FloatField()
|
|
model = CharField(max_length=30)
|
|
embedding = BlobField()
|
|
triggering_event_id = CharField(max_length=30)
|
|
last_triggered = DateTimeField()
|
|
|
|
class Meta:
|
|
primary_key = CompositeKey("camera", "name")
|