2024-10-07 22:30:45 +02:00
|
|
|
"""SQLite-vec embeddings database."""
|
2024-06-21 23:30:19 +02:00
|
|
|
|
|
|
|
import base64
|
|
|
|
import io
|
|
|
|
import logging
|
2024-10-07 22:30:45 +02:00
|
|
|
import struct
|
2024-06-21 23:30:19 +02:00
|
|
|
import time
|
2024-10-07 22:30:45 +02:00
|
|
|
from typing import List, Tuple, Union
|
2024-06-21 23:30:19 +02:00
|
|
|
|
2024-10-09 23:31:54 +02:00
|
|
|
import numpy as np
|
2024-06-21 23:30:19 +02:00
|
|
|
from PIL import Image
|
|
|
|
from playhouse.shortcuts import model_to_dict
|
|
|
|
|
2024-10-07 22:30:45 +02:00
|
|
|
from frigate.comms.inter_process import InterProcessRequestor
|
2024-10-10 15:09:12 +02:00
|
|
|
from frigate.config.semantic_search import SemanticSearchConfig
|
2024-10-07 22:30:45 +02:00
|
|
|
from frigate.const import UPDATE_MODEL_STATE
|
|
|
|
from frigate.db.sqlitevecq import SqliteVecQueueDatabase
|
2024-06-21 23:30:19 +02:00
|
|
|
from frigate.models import Event
|
2024-10-07 22:30:45 +02:00
|
|
|
from frigate.types import ModelStatusTypesEnum
|
2024-06-21 23:30:19 +02:00
|
|
|
|
2024-10-09 23:31:54 +02:00
|
|
|
from .functions.onnx import GenericONNXEmbedding
|
2024-06-21 23:30:19 +02:00
|
|
|
|
|
|
|
logger = logging.getLogger(__name__)
|
|
|
|
|
|
|
|
|
|
|
|
def get_metadata(event: Event) -> dict:
|
|
|
|
"""Extract valid event metadata."""
|
|
|
|
event_dict = model_to_dict(event)
|
|
|
|
return (
|
|
|
|
{
|
|
|
|
k: v
|
|
|
|
for k, v in event_dict.items()
|
2024-09-26 22:30:56 +02:00
|
|
|
if k not in ["thumbnail"]
|
2024-06-21 23:30:19 +02:00
|
|
|
and v is not None
|
|
|
|
and isinstance(v, (str, int, float, bool))
|
|
|
|
}
|
|
|
|
| {
|
|
|
|
k: v
|
|
|
|
for k, v in event_dict["data"].items()
|
|
|
|
if k not in ["description"]
|
|
|
|
and v is not None
|
|
|
|
and isinstance(v, (str, int, float, bool))
|
|
|
|
}
|
|
|
|
| {
|
|
|
|
# Metadata search doesn't support $contains
|
|
|
|
# and an event can have multiple zones, so
|
|
|
|
# we need to create a key for each zone
|
|
|
|
f"{k}_{x}": True
|
|
|
|
for k, v in event_dict.items()
|
|
|
|
if isinstance(v, list) and len(v) > 0
|
|
|
|
for x in v
|
|
|
|
if isinstance(x, str)
|
|
|
|
}
|
|
|
|
)
|
|
|
|
|
|
|
|
|
2024-10-09 23:31:54 +02:00
|
|
|
def serialize(vector: Union[List[float], np.ndarray, float]) -> bytes:
|
|
|
|
"""Serializes a list of floats, numpy array, or single float into a compact "raw bytes" format"""
|
|
|
|
if isinstance(vector, np.ndarray):
|
|
|
|
# Convert numpy array to list of floats
|
|
|
|
vector = vector.flatten().tolist()
|
|
|
|
elif isinstance(vector, (float, np.float32, np.float64)):
|
|
|
|
# Handle single float values
|
|
|
|
vector = [vector]
|
|
|
|
elif not isinstance(vector, list):
|
|
|
|
raise TypeError(
|
|
|
|
f"Input must be a list of floats, a numpy array, or a single float. Got {type(vector)}"
|
|
|
|
)
|
|
|
|
|
|
|
|
try:
|
|
|
|
return struct.pack("%sf" % len(vector), *vector)
|
|
|
|
except struct.error as e:
|
|
|
|
raise ValueError(f"Failed to pack vector: {e}. Vector: {vector}")
|
2024-10-07 22:30:45 +02:00
|
|
|
|
|
|
|
|
|
|
|
def deserialize(bytes_data: bytes) -> List[float]:
|
|
|
|
"""Deserializes a compact "raw bytes" format into a list of floats"""
|
|
|
|
return list(struct.unpack("%sf" % (len(bytes_data) // 4), bytes_data))
|
|
|
|
|
|
|
|
|
2024-06-21 23:30:19 +02:00
|
|
|
class Embeddings:
|
2024-10-07 22:30:45 +02:00
|
|
|
"""SQLite-vec embeddings database."""
|
|
|
|
|
2024-10-10 15:09:12 +02:00
|
|
|
def __init__(
|
|
|
|
self, config: SemanticSearchConfig, db: SqliteVecQueueDatabase
|
|
|
|
) -> None:
|
|
|
|
self.config = config
|
2024-10-07 22:30:45 +02:00
|
|
|
self.db = db
|
|
|
|
self.requestor = InterProcessRequestor()
|
|
|
|
|
|
|
|
# Create tables if they don't exist
|
|
|
|
self._create_tables()
|
|
|
|
|
|
|
|
models = [
|
2024-10-09 23:31:54 +02:00
|
|
|
"jinaai/jina-clip-v1-text_model_fp16.onnx",
|
|
|
|
"jinaai/jina-clip-v1-tokenizer",
|
|
|
|
"jinaai/jina-clip-v1-vision_model_fp16.onnx",
|
|
|
|
"jinaai/jina-clip-v1-preprocessor_config.json",
|
2024-10-07 22:30:45 +02:00
|
|
|
]
|
|
|
|
|
|
|
|
for model in models:
|
|
|
|
self.requestor.send_data(
|
|
|
|
UPDATE_MODEL_STATE,
|
|
|
|
{
|
|
|
|
"model": model,
|
|
|
|
"state": ModelStatusTypesEnum.not_downloaded,
|
|
|
|
},
|
|
|
|
)
|
|
|
|
|
2024-10-09 23:31:54 +02:00
|
|
|
def jina_text_embedding_function(outputs):
|
|
|
|
return outputs[0]
|
|
|
|
|
|
|
|
def jina_vision_embedding_function(outputs):
|
|
|
|
return outputs[0]
|
|
|
|
|
|
|
|
self.text_embedding = GenericONNXEmbedding(
|
|
|
|
model_name="jinaai/jina-clip-v1",
|
|
|
|
model_file="text_model_fp16.onnx",
|
|
|
|
tokenizer_file="tokenizer",
|
|
|
|
download_urls={
|
|
|
|
"text_model_fp16.onnx": "https://huggingface.co/jinaai/jina-clip-v1/resolve/main/onnx/text_model_fp16.onnx",
|
|
|
|
},
|
|
|
|
embedding_function=jina_text_embedding_function,
|
|
|
|
model_type="text",
|
2024-10-10 15:09:12 +02:00
|
|
|
device="CPU",
|
2024-10-07 22:30:45 +02:00
|
|
|
)
|
2024-10-09 23:31:54 +02:00
|
|
|
|
|
|
|
self.vision_embedding = GenericONNXEmbedding(
|
|
|
|
model_name="jinaai/jina-clip-v1",
|
|
|
|
model_file="vision_model_fp16.onnx",
|
|
|
|
download_urls={
|
|
|
|
"vision_model_fp16.onnx": "https://huggingface.co/jinaai/jina-clip-v1/resolve/main/onnx/vision_model_fp16.onnx",
|
|
|
|
"preprocessor_config.json": "https://huggingface.co/jinaai/jina-clip-v1/resolve/main/preprocessor_config.json",
|
|
|
|
},
|
|
|
|
embedding_function=jina_vision_embedding_function,
|
|
|
|
model_type="vision",
|
2024-10-10 15:09:12 +02:00
|
|
|
device=self.config.device,
|
2024-10-07 22:30:45 +02:00
|
|
|
)
|
|
|
|
|
|
|
|
def _create_tables(self):
|
|
|
|
# Create vec0 virtual table for thumbnail embeddings
|
|
|
|
self.db.execute_sql("""
|
|
|
|
CREATE VIRTUAL TABLE IF NOT EXISTS vec_thumbnails USING vec0(
|
|
|
|
id TEXT PRIMARY KEY,
|
2024-10-09 23:31:54 +02:00
|
|
|
thumbnail_embedding FLOAT[768]
|
2024-10-07 22:30:45 +02:00
|
|
|
);
|
|
|
|
""")
|
|
|
|
|
|
|
|
# Create vec0 virtual table for description embeddings
|
|
|
|
self.db.execute_sql("""
|
|
|
|
CREATE VIRTUAL TABLE IF NOT EXISTS vec_descriptions USING vec0(
|
|
|
|
id TEXT PRIMARY KEY,
|
2024-10-09 23:31:54 +02:00
|
|
|
description_embedding FLOAT[768]
|
2024-10-07 22:30:45 +02:00
|
|
|
);
|
|
|
|
""")
|
|
|
|
|
2024-10-09 23:31:54 +02:00
|
|
|
def _drop_tables(self):
|
|
|
|
self.db.execute_sql("""
|
|
|
|
DROP TABLE vec_descriptions;
|
|
|
|
""")
|
|
|
|
self.db.execute_sql("""
|
|
|
|
DROP TABLE vec_thumbnails;
|
|
|
|
""")
|
|
|
|
|
2024-10-07 22:30:45 +02:00
|
|
|
def upsert_thumbnail(self, event_id: str, thumbnail: bytes):
|
|
|
|
# Convert thumbnail bytes to PIL Image
|
|
|
|
image = Image.open(io.BytesIO(thumbnail)).convert("RGB")
|
2024-10-09 23:31:54 +02:00
|
|
|
embedding = self.vision_embedding([image])[0]
|
2024-10-07 22:30:45 +02:00
|
|
|
|
|
|
|
self.db.execute_sql(
|
|
|
|
"""
|
|
|
|
INSERT OR REPLACE INTO vec_thumbnails(id, thumbnail_embedding)
|
|
|
|
VALUES(?, ?)
|
|
|
|
""",
|
|
|
|
(event_id, serialize(embedding)),
|
|
|
|
)
|
|
|
|
|
|
|
|
return embedding
|
|
|
|
|
|
|
|
def upsert_description(self, event_id: str, description: str):
|
2024-10-09 23:31:54 +02:00
|
|
|
embedding = self.text_embedding([description])[0]
|
2024-10-07 22:30:45 +02:00
|
|
|
|
|
|
|
self.db.execute_sql(
|
|
|
|
"""
|
|
|
|
INSERT OR REPLACE INTO vec_descriptions(id, description_embedding)
|
|
|
|
VALUES(?, ?)
|
|
|
|
""",
|
|
|
|
(event_id, serialize(embedding)),
|
|
|
|
)
|
|
|
|
|
|
|
|
return embedding
|
2024-06-21 23:30:19 +02:00
|
|
|
|
2024-10-07 22:30:45 +02:00
|
|
|
def delete_thumbnail(self, event_ids: List[str]) -> None:
|
|
|
|
ids = ",".join(["?" for _ in event_ids])
|
|
|
|
self.db.execute_sql(
|
|
|
|
f"DELETE FROM vec_thumbnails WHERE id IN ({ids})", event_ids
|
2024-06-21 23:30:19 +02:00
|
|
|
)
|
|
|
|
|
2024-10-07 22:30:45 +02:00
|
|
|
def delete_description(self, event_ids: List[str]) -> None:
|
|
|
|
ids = ",".join(["?" for _ in event_ids])
|
|
|
|
self.db.execute_sql(
|
|
|
|
f"DELETE FROM vec_descriptions WHERE id IN ({ids})", event_ids
|
2024-06-21 23:30:19 +02:00
|
|
|
)
|
|
|
|
|
2024-10-07 22:30:45 +02:00
|
|
|
def search_thumbnail(
|
|
|
|
self, query: Union[Event, str], event_ids: List[str] = None
|
|
|
|
) -> List[Tuple[str, float]]:
|
|
|
|
if query.__class__ == Event:
|
|
|
|
cursor = self.db.execute_sql(
|
|
|
|
"""
|
|
|
|
SELECT thumbnail_embedding FROM vec_thumbnails WHERE id = ?
|
|
|
|
""",
|
|
|
|
[query.id],
|
|
|
|
)
|
|
|
|
|
|
|
|
row = cursor.fetchone() if cursor else None
|
|
|
|
|
|
|
|
if row:
|
|
|
|
query_embedding = deserialize(
|
|
|
|
row[0]
|
|
|
|
) # Deserialize the thumbnail embedding
|
|
|
|
else:
|
|
|
|
# If no embedding found, generate it and return it
|
|
|
|
thumbnail = base64.b64decode(query.thumbnail)
|
|
|
|
query_embedding = self.upsert_thumbnail(query.id, thumbnail)
|
|
|
|
else:
|
2024-10-09 23:31:54 +02:00
|
|
|
query_embedding = self.text_embedding([query])[0]
|
2024-10-07 22:30:45 +02:00
|
|
|
|
|
|
|
sql_query = """
|
|
|
|
SELECT
|
|
|
|
id,
|
|
|
|
distance
|
|
|
|
FROM vec_thumbnails
|
|
|
|
WHERE thumbnail_embedding MATCH ?
|
|
|
|
AND k = 100
|
|
|
|
"""
|
|
|
|
|
|
|
|
# Add the IN clause if event_ids is provided and not empty
|
|
|
|
# this is the only filter supported by sqlite-vec as of 0.1.3
|
|
|
|
# but it seems to be broken in this version
|
|
|
|
if event_ids:
|
|
|
|
sql_query += " AND id IN ({})".format(",".join("?" * len(event_ids)))
|
|
|
|
|
|
|
|
# order by distance DESC is not implemented in this version of sqlite-vec
|
|
|
|
# when it's implemented, we can use cosine similarity
|
|
|
|
sql_query += " ORDER BY distance"
|
|
|
|
|
|
|
|
parameters = (
|
|
|
|
[serialize(query_embedding)] + event_ids
|
|
|
|
if event_ids
|
|
|
|
else [serialize(query_embedding)]
|
2024-06-21 23:30:19 +02:00
|
|
|
)
|
|
|
|
|
2024-10-07 22:30:45 +02:00
|
|
|
results = self.db.execute_sql(sql_query, parameters).fetchall()
|
|
|
|
|
|
|
|
return results
|
|
|
|
|
|
|
|
def search_description(
|
|
|
|
self, query_text: str, event_ids: List[str] = None
|
|
|
|
) -> List[Tuple[str, float]]:
|
2024-10-09 23:31:54 +02:00
|
|
|
query_embedding = self.text_embedding([query_text])[0]
|
2024-10-07 22:30:45 +02:00
|
|
|
|
|
|
|
# Prepare the base SQL query
|
|
|
|
sql_query = """
|
|
|
|
SELECT
|
|
|
|
id,
|
|
|
|
distance
|
|
|
|
FROM vec_descriptions
|
|
|
|
WHERE description_embedding MATCH ?
|
|
|
|
AND k = 100
|
|
|
|
"""
|
|
|
|
|
|
|
|
# Add the IN clause if event_ids is provided and not empty
|
|
|
|
# this is the only filter supported by sqlite-vec as of 0.1.3
|
|
|
|
# but it seems to be broken in this version
|
|
|
|
if event_ids:
|
|
|
|
sql_query += " AND id IN ({})".format(",".join("?" * len(event_ids)))
|
|
|
|
|
|
|
|
# order by distance DESC is not implemented in this version of sqlite-vec
|
|
|
|
# when it's implemented, we can use cosine similarity
|
|
|
|
sql_query += " ORDER BY distance"
|
|
|
|
|
|
|
|
parameters = (
|
|
|
|
[serialize(query_embedding)] + event_ids
|
|
|
|
if event_ids
|
|
|
|
else [serialize(query_embedding)]
|
|
|
|
)
|
|
|
|
|
|
|
|
results = self.db.execute_sql(sql_query, parameters).fetchall()
|
|
|
|
|
|
|
|
return results
|
|
|
|
|
2024-06-21 23:30:19 +02:00
|
|
|
def reindex(self) -> None:
|
|
|
|
logger.info("Indexing event embeddings...")
|
|
|
|
|
2024-10-09 23:31:54 +02:00
|
|
|
self._drop_tables()
|
|
|
|
self._create_tables()
|
|
|
|
|
2024-06-21 23:30:19 +02:00
|
|
|
st = time.time()
|
2024-06-23 21:27:21 +02:00
|
|
|
totals = {
|
|
|
|
"thumb": 0,
|
|
|
|
"desc": 0,
|
|
|
|
}
|
2024-06-21 23:30:19 +02:00
|
|
|
|
2024-06-23 21:27:21 +02:00
|
|
|
batch_size = 100
|
|
|
|
current_page = 1
|
|
|
|
events = (
|
|
|
|
Event.select()
|
|
|
|
.where(
|
|
|
|
(Event.has_clip == True | Event.has_snapshot == True)
|
|
|
|
& Event.thumbnail.is_null(False)
|
2024-06-21 23:30:19 +02:00
|
|
|
)
|
2024-06-23 21:27:21 +02:00
|
|
|
.order_by(Event.start_time.desc())
|
|
|
|
.paginate(current_page, batch_size)
|
|
|
|
)
|
2024-06-21 23:30:19 +02:00
|
|
|
|
2024-06-23 21:27:21 +02:00
|
|
|
while len(events) > 0:
|
|
|
|
event: Event
|
|
|
|
for event in events:
|
|
|
|
thumbnail = base64.b64decode(event.thumbnail)
|
2024-10-07 22:30:45 +02:00
|
|
|
self.upsert_thumbnail(event.id, thumbnail)
|
|
|
|
totals["thumb"] += 1
|
2024-09-23 14:53:19 +02:00
|
|
|
if description := event.data.get("description", "").strip():
|
2024-10-07 22:30:45 +02:00
|
|
|
totals["desc"] += 1
|
|
|
|
self.upsert_description(event.id, description)
|
2024-06-23 21:27:21 +02:00
|
|
|
|
|
|
|
current_page += 1
|
|
|
|
events = (
|
|
|
|
Event.select()
|
|
|
|
.where(
|
|
|
|
(Event.has_clip == True | Event.has_snapshot == True)
|
|
|
|
& Event.thumbnail.is_null(False)
|
|
|
|
)
|
|
|
|
.order_by(Event.start_time.desc())
|
|
|
|
.paginate(current_page, batch_size)
|
2024-06-21 23:30:19 +02:00
|
|
|
)
|
|
|
|
|
|
|
|
logger.info(
|
|
|
|
"Embedded %d thumbnails and %d descriptions in %s seconds",
|
2024-06-23 21:27:21 +02:00
|
|
|
totals["thumb"],
|
|
|
|
totals["desc"],
|
2024-06-21 23:30:19 +02:00
|
|
|
time.time() - st,
|
|
|
|
)
|