## Voice Pipeline (P3) - Replace openWakeWord daemon with Wyoming Satellite approach - Add Wyoming Satellite service on port 10700 for HA voice pipeline - Update setup.sh with cross-platform sed compatibility (macOS/Linux) - Add version field to Kokoro TTS voice info - Update launchd service loader to use Wyoming Satellite ## Home Assistant Integration (P4) - Add custom conversation agent component (openclaw_conversation) - Fix: Use IntentResponse instead of plain strings (HA API requirement) - Support both HTTP API and CLI fallback modes - Config flow for easy HA UI setup - Add OpenClaw bridge scripts (Python + Bash) - Add ha-ctl utility for HA entity control - Fix: Use context manager for token file reading - Add HA configuration examples and documentation ## Infrastructure - Add mem0 backup automation (launchd + script) - Add n8n workflow templates (morning briefing, notification router) - Add VS Code workspace configuration - Reorganize model files into categorized folders: - lmstudio-community/ - mlx-community/ - bartowski/ - mradermacher/ ## Documentation - Update PROJECT_PLAN.md with Wyoming Satellite architecture - Update TODO.md with completed Wyoming integration tasks - Add OPENCLAW_INTEGRATION.md for HA setup guide ## Testing - Verified Wyoming services running (STT:10300, TTS:10301, Satellite:10700) - Verified OpenClaw CLI accessibility - Confirmed cross-platform compatibility fixes
121 lines
5.2 KiB
Django/Jinja
121 lines
5.2 KiB
Django/Jinja
{%- if tools %}
|
|
{{- '<|im_start|>system\n' }}
|
|
{%- if messages[0].role == 'system' %}
|
|
{%- if messages[0].content is string %}
|
|
{{- messages[0].content }}
|
|
{%- else %}
|
|
{%- for content in messages[0].content %}
|
|
{%- if 'text' in content %}
|
|
{{- content.text }}
|
|
{%- endif %}
|
|
{%- endfor %}
|
|
{%- endif %}
|
|
{{- '\n\n' }}
|
|
{%- endif %}
|
|
{{- "# Tools\n\nYou may call one or more functions to assist with the user query.\n\nYou are provided with function signatures within <tools></tools> XML tags:\n<tools>" }}
|
|
{%- for tool in tools %}
|
|
{{- "\n" }}
|
|
{{- tool | tojson }}
|
|
{%- endfor %}
|
|
{{- "\n</tools>\n\nFor each function call, return a json object with function name and arguments within <tool_call></tool_call> XML tags:\n<tool_call>\n{\"name\": <function-name>, \"arguments\": <args-json-object>}\n</tool_call><|im_end|>\n" }}
|
|
{%- else %}
|
|
{%- if messages[0].role == 'system' %}
|
|
{{- '<|im_start|>system\n' }}
|
|
{%- if messages[0].content is string %}
|
|
{{- messages[0].content }}
|
|
{%- else %}
|
|
{%- for content in messages[0].content %}
|
|
{%- if 'text' in content %}
|
|
{{- content.text }}
|
|
{%- endif %}
|
|
{%- endfor %}
|
|
{%- endif %}
|
|
{{- '<|im_end|>\n' }}
|
|
{%- endif %}
|
|
{%- endif %}
|
|
{%- set image_count = namespace(value=0) %}
|
|
{%- set video_count = namespace(value=0) %}
|
|
{%- for message in messages %}
|
|
{%- if message.role == "user" %}
|
|
{{- '<|im_start|>' + message.role + '\n' }}
|
|
{%- if message.content is string %}
|
|
{{- message.content }}
|
|
{%- else %}
|
|
{%- for content in message.content %}
|
|
{%- if content.type == 'image' or 'image' in content or 'image_url' in content %}
|
|
{%- set image_count.value = image_count.value + 1 %}
|
|
{%- if add_vision_id %}Picture {{ image_count.value }}: {% endif -%}
|
|
<|vision_start|><|image_pad|><|vision_end|>
|
|
{%- elif content.type == 'video' or 'video' in content %}
|
|
{%- set video_count.value = video_count.value + 1 %}
|
|
{%- if add_vision_id %}Video {{ video_count.value }}: {% endif -%}
|
|
<|vision_start|><|video_pad|><|vision_end|>
|
|
{%- elif 'text' in content %}
|
|
{{- content.text }}
|
|
{%- endif %}
|
|
{%- endfor %}
|
|
{%- endif %}
|
|
{{- '<|im_end|>\n' }}
|
|
{%- elif message.role == "assistant" %}
|
|
{{- '<|im_start|>' + message.role + '\n' }}
|
|
{%- if message.content is string %}
|
|
{{- message.content }}
|
|
{%- else %}
|
|
{%- for content_item in message.content %}
|
|
{%- if 'text' in content_item %}
|
|
{{- content_item.text }}
|
|
{%- endif %}
|
|
{%- endfor %}
|
|
{%- endif %}
|
|
{%- if message.tool_calls %}
|
|
{%- for tool_call in message.tool_calls %}
|
|
{%- if (loop.first and message.content) or (not loop.first) %}
|
|
{{- '\n' }}
|
|
{%- endif %}
|
|
{%- if tool_call.function %}
|
|
{%- set tool_call = tool_call.function %}
|
|
{%- endif %}
|
|
{{- '<tool_call>\n{"name": "' }}
|
|
{{- tool_call.name }}
|
|
{{- '", "arguments": ' }}
|
|
{%- if tool_call.arguments is string %}
|
|
{{- tool_call.arguments }}
|
|
{%- else %}
|
|
{{- tool_call.arguments | tojson }}
|
|
{%- endif %}
|
|
{{- '}\n</tool_call>' }}
|
|
{%- endfor %}
|
|
{%- endif %}
|
|
{{- '<|im_end|>\n' }}
|
|
{%- elif message.role == "tool" %}
|
|
{%- if loop.first or (messages[loop.index0 - 1].role != "tool") %}
|
|
{{- '<|im_start|>user' }}
|
|
{%- endif %}
|
|
{{- '\n<tool_response>\n' }}
|
|
{%- if message.content is string %}
|
|
{{- message.content }}
|
|
{%- else %}
|
|
{%- for content in message.content %}
|
|
{%- if content.type == 'image' or 'image' in content or 'image_url' in content %}
|
|
{%- set image_count.value = image_count.value + 1 %}
|
|
{%- if add_vision_id %}Picture {{ image_count.value }}: {% endif -%}
|
|
<|vision_start|><|image_pad|><|vision_end|>
|
|
{%- elif content.type == 'video' or 'video' in content %}
|
|
{%- set video_count.value = video_count.value + 1 %}
|
|
{%- if add_vision_id %}Video {{ video_count.value }}: {% endif -%}
|
|
<|vision_start|><|video_pad|><|vision_end|>
|
|
{%- elif 'text' in content %}
|
|
{{- content.text }}
|
|
{%- endif %}
|
|
{%- endfor %}
|
|
{%- endif %}
|
|
{{- '\n</tool_response>' }}
|
|
{%- if loop.last or (messages[loop.index0 + 1].role != "tool") %}
|
|
{{- '<|im_end|>\n' }}
|
|
{%- endif %}
|
|
{%- endif %}
|
|
{%- endfor %}
|
|
{%- if add_generation_prompt %}
|
|
{{- '<|im_start|>assistant\n' }}
|
|
{%- endif %}
|