Files
homeai/homeai-agent/custom_components/openclaw_conversation/conversation.py
Aodhan Collins 664bb6d275 feat: OpenClaw HTTP bridge, HA conversation agent fixes, voice pipeline tooling
- Add openclaw-http-bridge.py: HTTP server translating POST requests to OpenClaw CLI calls
- Add launchd plist for HTTP bridge (port 8081, auto-start)
- Add install-to-docker-ha.sh: deploy custom component to Docker HA via SSH
- Add package-for-ha.sh: create distributable tarball of custom component
- Add test-services.sh: comprehensive voice pipeline service checker

Fixes from code review:
- Use OpenClawAgent (HTTP) in async_setup_entry instead of OpenClawCLIAgent
  (CLI agent fails inside Docker HA where openclaw binary doesn't exist)
- Update all port references from 8080 to 8081 (HTTP bridge port)
- Remove overly permissive CORS headers from HTTP bridge
- Fix zombie process leak: kill child process on CLI timeout
- Remove unused subprocess import in conversation.py
- Add version field to Kokoro TTS Wyoming info
- Update TODO.md with voice pipeline progress
2026-03-08 22:46:04 +00:00

225 lines
7.8 KiB
Python

"""Conversation agent for OpenClaw integration."""
from __future__ import annotations
import asyncio
import json
import logging
from typing import Any
import aiohttp
import voluptuous as vol
from homeassistant.components.conversation import (
AbstractConversationAgent,
ConversationInput,
ConversationResult,
)
from homeassistant.core import HomeAssistant
from homeassistant.helpers.aiohttp_client import async_get_clientsession
from homeassistant.helpers.intent import IntentResponse
from .const import (
CONF_AGENT_NAME,
CONF_OPENCLAW_HOST,
CONF_OPENCLAW_PORT,
CONF_TIMEOUT,
DEFAULT_AGENT,
DEFAULT_HOST,
DEFAULT_PORT,
DEFAULT_TIMEOUT,
DOMAIN,
OPENCLAW_API_PATH,
)
_LOGGER = logging.getLogger(__name__)
# Schema for configuration
CONFIG_SCHEMA = vol.Schema({
vol.Optional(CONF_OPENCLAW_HOST, default=DEFAULT_HOST): str,
vol.Optional(CONF_OPENCLAW_PORT, default=DEFAULT_PORT): int,
vol.Optional(CONF_AGENT_NAME, default=DEFAULT_AGENT): str,
vol.Optional(CONF_TIMEOUT, default=DEFAULT_TIMEOUT): int,
})
class OpenClawAgent(AbstractConversationAgent):
"""OpenClaw conversation agent."""
def __init__(self, hass: HomeAssistant, config: dict[str, Any]) -> None:
"""Initialize the agent."""
self.hass = hass
self.config = config
self.host = config.get(CONF_OPENCLAW_HOST, DEFAULT_HOST)
self.port = config.get(CONF_OPENCLAW_PORT, DEFAULT_PORT)
self.agent_name = config.get(CONF_AGENT_NAME, DEFAULT_AGENT)
self.timeout = config.get(CONF_TIMEOUT, DEFAULT_TIMEOUT)
@property
def supported_languages(self) -> list[str]:
"""Return a list of supported languages."""
return ["en"] # OpenClaw primarily supports English
@property
def attribution(self) -> dict[str, str] | None:
"""Return attribution information."""
return {
"name": "OpenClaw",
"url": "https://github.com/homeai/openclaw",
}
async def async_process(
self, user_input: ConversationInput
) -> ConversationResult:
"""Process a sentence and return a response."""
text = user_input.text
conversation_id = user_input.conversation_id
_LOGGER.debug("Processing message: %s", text)
try:
response_text = await self._call_openclaw(text)
# Create proper IntentResponse for Home Assistant
intent_response = IntentResponse(language=user_input.language or "en")
intent_response.async_set_speech(response_text)
return ConversationResult(
response=intent_response,
conversation_id=conversation_id,
)
except Exception as err:
_LOGGER.error("Error calling OpenClaw: %s", err)
intent_response = IntentResponse(language=user_input.language or "en")
intent_response.async_set_speech("I'm sorry, I encountered an error processing your request.")
return ConversationResult(
response=intent_response,
conversation_id=conversation_id,
)
async def _call_openclaw(self, message: str) -> str:
"""Call OpenClaw API and return the response."""
url = f"http://{self.host}:{self.port}{OPENCLAW_API_PATH}"
payload = {
"message": message,
"agent": self.agent_name,
}
session = async_get_clientsession(self.hass)
try:
async with asyncio.timeout(self.timeout):
async with session.post(
url,
json=payload,
headers={"Content-Type": "application/json"},
) as response:
if response.status != 200:
_LOGGER.error(
"OpenClaw returned status %s: %s",
response.status,
await response.text(),
)
return "I'm sorry, I couldn't process that request."
data = await response.json()
return data.get("response", "I didn't get a response.")
except asyncio.TimeoutError:
_LOGGER.error("Timeout calling OpenClaw")
return "I'm sorry, the request timed out."
except aiohttp.ClientError as err:
_LOGGER.error("Error connecting to OpenClaw: %s", err)
return "I'm sorry, I couldn't connect to the OpenClaw service."
class OpenClawCLIAgent(AbstractConversationAgent):
"""OpenClaw conversation agent using CLI (fallback if HTTP API unavailable)."""
def __init__(self, hass: HomeAssistant, config: dict[str, Any]) -> None:
"""Initialize the agent."""
self.hass = hass
self.config = config
self.agent_name = config.get(CONF_AGENT_NAME, DEFAULT_AGENT)
self.timeout = config.get(CONF_TIMEOUT, DEFAULT_TIMEOUT)
@property
def supported_languages(self) -> list[str]:
"""Return a list of supported languages."""
return ["en"]
@property
def attribution(self) -> dict[str, str] | None:
"""Return attribution information."""
return {
"name": "OpenClaw",
"url": "https://github.com/homeai/openclaw",
}
async def async_process(
self, user_input: ConversationInput
) -> ConversationResult:
"""Process a sentence using OpenClaw CLI."""
text = user_input.text
conversation_id = user_input.conversation_id
_LOGGER.debug("Processing message via CLI: %s", text)
try:
response_text = await self._call_openclaw_cli(text)
# Create proper IntentResponse for Home Assistant
intent_response = IntentResponse(language=user_input.language or "en")
intent_response.async_set_speech(response_text)
return ConversationResult(
response=intent_response,
conversation_id=conversation_id,
)
except Exception as err:
_LOGGER.error("Error calling OpenClaw CLI: %s", err)
intent_response = IntentResponse(language=user_input.language or "en")
intent_response.async_set_speech("I'm sorry, I encountered an error processing your request.")
return ConversationResult(
response=intent_response,
conversation_id=conversation_id,
)
async def _call_openclaw_cli(self, message: str) -> str:
"""Call OpenClaw CLI and return the response."""
cmd = [
"openclaw",
"agent",
"--message", message,
"--agent", self.agent_name,
]
proc = None
try:
proc = await asyncio.create_subprocess_exec(
*cmd,
stdout=asyncio.subprocess.PIPE,
stderr=asyncio.subprocess.PIPE,
)
stdout, stderr = await asyncio.wait_for(
proc.communicate(),
timeout=self.timeout,
)
if proc.returncode != 0:
_LOGGER.error("OpenClaw CLI failed: %s", stderr.decode().strip())
return "I'm sorry, I couldn't process that request."
return stdout.decode().strip()
except asyncio.TimeoutError:
if proc is not None:
proc.kill()
await proc.wait()
_LOGGER.error("Timeout calling OpenClaw CLI")
return "I'm sorry, the request timed out."
except FileNotFoundError:
_LOGGER.error("OpenClaw CLI not found")
return "I'm sorry, OpenClaw is not available."