## 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
166 lines
7.6 KiB
Django/Jinja
166 lines
7.6 KiB
Django/Jinja
{# ----------‑‑‑ special token variables ‑‑‑---------- #}
|
||
{%- set bos_token = '<seed:bos>' -%}
|
||
{%- set eos_token = '<seed:eos>' -%}
|
||
{%- set pad_token = '<seed:pad>' -%}
|
||
{%- set toolcall_begin_token = '<seed:tool_call>' -%}
|
||
{%- set toolcall_end_token = '</seed:tool_call>' -%}
|
||
{%- set think_begin_token = '<seed:think>' -%}
|
||
{%- set think_end_token = '</seed:think>' -%}
|
||
{%- set budget_begin_token = '<seed:cot_budget_reflect>'-%}
|
||
{%- set budget_end_token = '</seed:cot_budget_reflect>'-%}
|
||
{# -------------- reflection-interval lookup -------------- #}
|
||
{%- if not thinking_budget is defined %}
|
||
{%- set thinking_budget = -1 -%}
|
||
{%- else -%}
|
||
{%- set thinking_budget = thinking_budget | int -%}
|
||
{%- endif -%}
|
||
{%- set budget_keys_v05 = [0, 512, 1024, 2048, 4096, 8192, 16384] -%}
|
||
{%- set budget_values_v05 = [0, 128, 256, 512, 512, 1024, 1024] -%}
|
||
{# 找到 "大于等于 thinking_budget" 的第一个档位 #}
|
||
{%- set ns = namespace(interval = None) -%}
|
||
{%- for i in range(budget_keys_v05|length) -%}
|
||
{%- if ns.interval is none and thinking_budget <= budget_keys_v05[i] -%}
|
||
{%- set ns.interval = budget_values_v05[i] -%}
|
||
{%- endif -%}
|
||
{%- endfor -%}
|
||
{# Find the first gear that is greater than or equal to the thinking_budget. #}
|
||
{%- if ns.interval is none -%}
|
||
{%- set ns.interval = budget_values_v05[-1] -%}
|
||
{%- endif -%}
|
||
{# ---------- Preprocess the system message ---------- #}
|
||
{%- if messages[0]["role"] == "system" %}
|
||
{%- set system_message = messages[0]["content"] %}
|
||
{%- set loop_messages = messages[1:] %}
|
||
{%- else %}
|
||
{%- set loop_messages = messages %}
|
||
{%- endif %}
|
||
{# ---------- Ensure tools exist ---------- #}
|
||
{%- if not tools is defined or tools is none %}
|
||
{%- set tools = [] %}
|
||
{%- endif %}
|
||
{# tools2doc.jinja #}
|
||
{%- macro py_type(t) -%}
|
||
{%- if t == "string" -%}str
|
||
{%- elif t in ("number", "integer") -%}int
|
||
{%- elif t == "boolean" -%}bool
|
||
{%- elif t == "array" -%}list
|
||
{%- else -%}Any{%- endif -%}
|
||
{%- endmacro -%}
|
||
{# ---------- Output the system block ---------- #}
|
||
{%- if system_message is defined %}
|
||
{{ bos_token + "system\n" + system_message }}
|
||
{%- else %}
|
||
{%- if tools is iterable and tools | length > 0 %}
|
||
{{ bos_token + "system\nYou are Doubao, a helpful AI assistant. You may call one or more functions to assist with the user query." }}
|
||
{%- endif %}
|
||
{%- endif %}
|
||
{%- if use_json_tooldef is defined and use_json_tooldef %}
|
||
|
||
{{"Tool List:\nYou are authorized to use the following tools (described in JSON Schema format). Before performing any task, you must decide how to call them based on the descriptions and parameters of these tools."}}
|
||
{{ tools | tojson(ensure_ascii=False) }}
|
||
{%- else %}
|
||
{%- for item in tools if item.type == "function" %}
|
||
|
||
|
||
Function:
|
||
def {{ item.function.name }}(
|
||
{%- for name, spec in item.function.parameters.properties.items() %}
|
||
{{- name }}: {{ py_type(spec.type) }}{% if not loop.last %},{% endif %}
|
||
{%- endfor %}):
|
||
"""
|
||
{{ item.function.description | trim }}
|
||
|
||
{# ---------- Args ---------- #}
|
||
{%- if item.function.parameters.properties %}
|
||
Args:
|
||
{%- for name, spec in item.function.parameters.properties.items() %}
|
||
|
||
- {{ name }} ({{ py_type(spec.type) }})
|
||
{%- if name in item.function.parameters.required %} [必填]{% else %} [选填]{% endif %}:
|
||
{{- " " ~ (spec.description or "") }}
|
||
{%- endfor %}
|
||
{%- endif %}
|
||
|
||
{# ---------- Returns ---------- #}
|
||
{%- if item.function.returns is defined
|
||
and item.function.returns.properties is defined
|
||
and item.function.returns.properties %}
|
||
Returns:
|
||
{%- for name, spec in item.function.returns.properties.items() %}
|
||
|
||
- {{ name }} ({{ py_type(spec.type) }}):
|
||
{{- " " ~ (spec.description or "") }}
|
||
{%- endfor %}
|
||
{%- endif %}
|
||
|
||
"""
|
||
{%- endfor %}
|
||
{%- endif %}
|
||
{%- if tools is iterable and tools | length > 0 %}
|
||
|
||
{{"工具调用请遵循如下格式:\n<seed:tool_call>\n<function=example_function_name>\n<parameter=example_parameter_1>value_1</parameter>\n<parameter=example_parameter_2>This is the value for the second parameter\nthat can span\nmultiple lines</parameter>\n</function>\n</seed:tool_call>\n"}}
|
||
{%- endif %}
|
||
{# End the system block line #}
|
||
{%- if system_message is defined or tools is iterable and tools | length > 0 %}
|
||
{{ eos_token }}
|
||
{%- endif %}
|
||
{# ---------- Thinking Budget ---------- #}
|
||
{%- if thinking_budget is defined %}
|
||
{%- if thinking_budget == 0 %}
|
||
{{ bos_token+"system" }}
|
||
{{ "You are an intelligent assistant that can answer questions in one step without the need for reasoning and thinking, that is, your thinking budget is 0. Next, please skip the thinking process and directly start answering the user's questions." }}
|
||
{{ eos_token }}
|
||
{%- elif not thinking_budget == -1 %}
|
||
{{ bos_token+"system" }}
|
||
{{ "You are an intelligent assistant with reflective ability. In the process of thinking and reasoning, you need to strictly follow the thinking budget, which is "}}{{thinking_budget}}{{". That is, you need to complete your thinking within "}}{{thinking_budget}}{{" tokens and start answering the user's questions. You will reflect on your thinking process every "}}{{ns.interval}}{{" tokens, stating how many tokens have been used and how many are left."}}
|
||
{{ eos_token }}
|
||
{%- endif %}
|
||
{%- endif %}
|
||
{# ---------- List the historical messages one by one ---------- #}
|
||
{%- for message in loop_messages %}
|
||
{%- if message.role == "assistant"
|
||
and message.tool_calls is defined
|
||
and message.tool_calls is iterable
|
||
and message.tool_calls | length > 0 %}
|
||
{{ bos_token + message.role }}
|
||
{%- if message.reasoning_content is defined and message.reasoning_content is string and message.reasoning_content | trim | length > 0 %}
|
||
{{ "\n" + think_begin_token + message.reasoning_content | trim + think_end_token }}
|
||
{%- endif %}
|
||
{%- if message.content is defined and message.content is string and message.content | trim | length > 0 %}
|
||
{{ "\n" + message.content | trim + "\n" }}
|
||
{%- endif %}
|
||
{%- for tool_call in message.tool_calls %}
|
||
{%- if tool_call.function is defined %}{% set tool_call = tool_call.function %}{% endif %}
|
||
{{ "\n" + toolcall_begin_token + "\n<function=" + tool_call.name + ">\n" }}
|
||
{%- if tool_call.arguments is defined %}
|
||
{%- for arg_name, arg_value in tool_call.arguments | items %}
|
||
{{ "<parameter=" + arg_name + ">" }}
|
||
{%- set arg_value = arg_value if arg_value is string else arg_value | string %}
|
||
{{ arg_value+"</parameter>\n" }}
|
||
{%- endfor %}
|
||
{%- endif %}
|
||
{{ "</function>\n" + toolcall_end_token }}
|
||
{%- endfor %}
|
||
{{ eos_token }}
|
||
{%- elif message.role in ["user", "system"] %}
|
||
{{ bos_token + message.role + "\n" + message.content + eos_token }}
|
||
{%- elif message.role == "assistant" %}
|
||
{{ bos_token + message.role }}
|
||
{%- if message.reasoning_content is defined and message.reasoning_content is string and message.reasoning_content | trim | length > 0 %}
|
||
{{ "\n" + think_begin_token + message.reasoning_content | trim + think_end_token }}
|
||
{%- endif %}
|
||
{%- if message.content is defined and message.content is string and message.content | trim | length > 0 %}
|
||
{{ "\n" + message.content | trim + eos_token }}
|
||
{%- endif %}
|
||
{# Include the tool role #}
|
||
{%- else %}
|
||
{{ bos_token + message.role + "\n" + message.content + eos_token }}
|
||
{%- endif %}
|
||
{%- endfor %}
|
||
{# ---------- Control the model to start continuation ---------- #}
|
||
{%- if add_generation_prompt %}
|
||
{{ bos_token+"assistant\n" }}
|
||
{%- if thinking_budget == 0 %}
|
||
{{ think_begin_token + "\n" + budget_begin_token + "The current thinking budget is 0, so I will directly start answering the question." + budget_end_token + "\n" + think_end_token }}
|
||
{%- endif %}
|
||
{%- endif %} |