mirror of
https://github.com/blakeblackshear/frigate.git
synced 2026-06-23 21:01:53 +03:00
* use monotonic clock for detector inference duration to prevent negative values from wall clock steps * add ability to set camera's webui_url from camera management pane * Gemini send thought signature * Update docs * copy face and lpr configs from source camera to replay camera * add guard * improve dummy camera docs * remove version number * fix stale field message after reverting a conditional form field Routes field-level conditional messages through a dedicated React Context instead of merging them into uiSchema. RJSF's Form keeps state.uiSchema sticky across renders during processPendingChange (formData is updated, uiSchema is not), so a previously injected ui:messages array stays attached to a field even after the triggering condition flips back to false. Context propagation re-runs FieldTemplate directly on every provider value change, sidestepping that staleness. * add semantic search field message to note that model_size is irrelevant when embeddings provider is selected --------- Co-authored-by: Nicolas Mowen <nickmowen213@gmail.com>
84 lines
2.8 KiB
Python
84 lines
2.8 KiB
Python
"""Shared helpers for GenAI providers and chat (OpenAI-style messages, tool call parsing)."""
|
|
|
|
import json
|
|
import logging
|
|
from typing import Any, List, Optional
|
|
|
|
logger = logging.getLogger(__name__)
|
|
|
|
|
|
def parse_tool_calls_from_message(
|
|
message: dict[str, Any],
|
|
) -> Optional[list[dict[str, Any]]]:
|
|
"""
|
|
Parse tool_calls from an OpenAI-style message dict.
|
|
|
|
Message may have "tool_calls" as a list of:
|
|
{"id": str, "function": {"name": str, "arguments": str}, ...}
|
|
|
|
Returns a list of {"id", "name", "arguments"} with arguments parsed as dict,
|
|
or None if no tool_calls. Used by Ollama and LlamaCpp (non-stream) responses.
|
|
"""
|
|
raw = message.get("tool_calls")
|
|
if not raw or not isinstance(raw, list):
|
|
return None
|
|
result = []
|
|
for idx, tool_call in enumerate(raw):
|
|
function_data = tool_call.get("function") or {}
|
|
raw_arguments = function_data.get("arguments") or {}
|
|
if isinstance(raw_arguments, dict):
|
|
arguments = raw_arguments
|
|
elif isinstance(raw_arguments, str):
|
|
try:
|
|
arguments = json.loads(raw_arguments)
|
|
except (json.JSONDecodeError, KeyError, TypeError) as e:
|
|
logger.warning(
|
|
"Failed to parse tool call arguments: %s, tool: %s",
|
|
e,
|
|
function_data.get("name", "unknown"),
|
|
)
|
|
arguments = {}
|
|
else:
|
|
arguments = {}
|
|
result.append(
|
|
{
|
|
"id": tool_call.get("id", "") or f"call_{idx}",
|
|
"name": function_data.get("name", ""),
|
|
"arguments": arguments,
|
|
}
|
|
)
|
|
return result if result else None
|
|
|
|
|
|
def build_assistant_message_for_conversation(
|
|
content: Any,
|
|
tool_calls_raw: Optional[List[dict[str, Any]]],
|
|
) -> dict[str, Any]:
|
|
"""
|
|
Build the assistant message dict in OpenAI format for appending to a conversation.
|
|
|
|
tool_calls_raw: list of {"id", "name", "arguments"} (arguments as dict), or None.
|
|
"""
|
|
msg: dict[str, Any] = {"role": "assistant", "content": content}
|
|
if tool_calls_raw:
|
|
msg["tool_calls"] = [
|
|
{
|
|
"id": tc["id"],
|
|
"type": "function",
|
|
"function": {
|
|
"name": tc["name"],
|
|
"arguments": json.dumps(tc.get("arguments") or {}),
|
|
},
|
|
# Gemini-only: opaque signature that must be echoed back on
|
|
# the same functionCall part in the next turn. Other providers
|
|
# do not set or read this.
|
|
**(
|
|
{"thought_signature": tc["thought_signature"]}
|
|
if tc.get("thought_signature")
|
|
else {}
|
|
),
|
|
}
|
|
for tc in tool_calls_raw
|
|
]
|
|
return msg
|