From 27aaaceff6c40ed6f5e0b25d1d474da4b85fd46a Mon Sep 17 00:00:00 2001 From: m50 Date: Mon, 16 Sep 2024 17:56:03 +0200 Subject: [PATCH] Fix #26: Implement Entity as HomeAssistant requires --- .../fallback_conversation/__init__.py | 169 +++------------ .../fallback_conversation/config_flow.py | 16 +- .../fallback_conversation/conversation.py | 196 ++++++++++++++++++ .../fallback_conversation/manifest.json | 3 +- 4 files changed, 230 insertions(+), 154 deletions(-) create mode 100644 custom_components/fallback_conversation/conversation.py diff --git a/custom_components/fallback_conversation/__init__.py b/custom_components/fallback_conversation/__init__.py index a496743..2ed16b1 100644 --- a/custom_components/fallback_conversation/__init__.py +++ b/custom_components/fallback_conversation/__init__.py @@ -3,167 +3,46 @@ import logging -from homeassistant.components import conversation from homeassistant.config_entries import ConfigEntry from homeassistant.core import HomeAssistant -from homeassistant.util import ulid -from home_assistant_intents import get_languages - - -from homeassistant.helpers import ( - config_validation as cv, - intent, -) - -from .const import ( - CONF_DEBUG_LEVEL, - CONF_PRIMARY_AGENT, - CONF_FALLBACK_AGENT, - DEBUG_LEVEL_NO_DEBUG, - DEBUG_LEVEL_LOW_DEBUG, - DEBUG_LEVEL_VERBOSE_DEBUG, - DOMAIN, - STRANGE_ERROR_RESPONSES, -) +from homeassistant.const import Platform +from homeassistant.helpers import config_validation as cv +from .const import DOMAIN _LOGGER = logging.getLogger(__name__) CONFIG_SCHEMA = cv.config_entry_only_config_schema(DOMAIN) +PLATFORMS = (Platform.CONVERSATION,) # hass.data key for agent. DATA_AGENT = "agent" - async def async_setup_entry(hass: HomeAssistant, entry: ConfigEntry) -> bool: """Set up Fallback Conversation Agent from a config entry.""" - agent = FallbackConversationAgent(hass, entry) - conversation.async_set_agent(hass, entry, agent) + hass.data.setdefault(DOMAIN, {})[entry.entry_id] = entry + + await hass.config_entries.async_forward_entry_setups(entry, PLATFORMS) return True +async def async_unload_entry(hass: HomeAssistant, entry: ConfigEntry) -> bool: + """Unload.""" + if not await hass.config_entries.async_unload_platforms(entry, PLATFORMS): + return False + hass.data[DOMAIN].pop(entry.entry_id) + return True + +async def async_migrate_entry(hass, config_entry: ConfigEntry): + """Migrate old entry.""" + _LOGGER.debug("Migrating from version %s", config_entry.version) + + if config_entry.version == 1: + _LOGGER.error("Cannot upgrade models that were created prior to v0.3. Please delete and re-create them.") + return False -class FallbackConversationAgent(conversation.AbstractConversationAgent): - """Fallback Conversation Agent.""" - - last_used_agent: str | None - - def __init__(self, hass: HomeAssistant, entry: ConfigEntry) -> None: - """Initialize the agent.""" - self.hass = hass - self.entry = entry - self.last_used_agent = None - - @property - def supported_languages(self) -> list[str]: - """Return a list of supported languages.""" - return get_languages() - - async def async_process( - self, user_input: conversation.ConversationInput - ) -> conversation.ConversationResult: - """Process a sentence.""" - agent_manager = conversation.get_agent_manager(self.hass) - default_agent = conversation.default_agent.async_get_default_agent(self.hass) - agent_names = self._convert_agent_info_to_dict( - agent_manager.async_get_agent_info() - ) - agent_names[conversation.const.HOME_ASSISTANT_AGENT] = default_agent.name - agent_names[conversation.const.OLD_HOME_ASSISTANT_AGENT] = default_agent.name - agents = [ - self.entry.options.get(CONF_PRIMARY_AGENT, default_agent), - self.entry.options.get(CONF_FALLBACK_AGENT, default_agent), - ] - - debug_level = self.entry.options.get(CONF_DEBUG_LEVEL, DEBUG_LEVEL_NO_DEBUG) - - if user_input.conversation_id is None: - user_input.conversation_id = ulid.ulid() - - all_results = [] - result = None - for agent_id in agents: - agent_name = "[unknown]" - if agent_id in agent_names: - agent_name = agent_names[agent_id] - else: - _LOGGER.warning("agent_name not found for agent_id %s", agent_id) - - result = await self._async_process_agent( - agent_manager, - agent_id, - agent_name, - user_input, - debug_level, - result, - ) - if result.response.response_type != intent.IntentResponseType.ERROR and result.response.speech['plain']['original_speech'].lower() not in STRANGE_ERROR_RESPONSES: - return result - all_results.append(result) - - intent_response = intent.IntentResponse(language=user_input.language) - err = "Complete fallback failure. No Conversation Agent was able to respond." - if debug_level == DEBUG_LEVEL_LOW_DEBUG: - r = all_results[-1].response.speech['plain'] - err += f"\n{r.get('agent_name', 'UNKNOWN')} responded with: {r.get('original_speech', r['speech'])}" - elif debug_level == DEBUG_LEVEL_VERBOSE_DEBUG: - for res in all_results: - r = res.response.speech['plain'] - err += f"\n{r.get('agent_name', 'UNKNOWN')} responded with: {r.get('original_speech', r['speech'])}" - intent_response.async_set_error( - intent.IntentResponseErrorCode.NO_INTENT_MATCH, - err, - ) - result = conversation.ConversationResult( - conversation_id=result.conversation_id, - response=intent_response - ) - - return result - - async def _async_process_agent( - self, - agent_manager: conversation.AgentManager, - agent_id: str, - agent_name: str, - user_input: conversation.ConversationInput, - debug_level: int, - previous_result, - ) -> conversation.ConversationResult: - """Process a specified agent.""" - agent = conversation.agent_manager.async_get_agent(self.hass, agent_id) - - _LOGGER.debug("Processing in %s using %s with debug level %s: %s", user_input.language, agent_id, debug_level, user_input.text) - - result = await agent.async_process(user_input) - r = result.response.speech['plain']['speech'] - result.response.speech['plain']['original_speech'] = r - result.response.speech['plain']['agent_name'] = agent_name - result.response.speech['plain']['agent_id'] = agent_id - if debug_level == DEBUG_LEVEL_LOW_DEBUG: - result.response.speech['plain']['speech'] = f"{agent_name} responded with: {r}" - elif debug_level == DEBUG_LEVEL_VERBOSE_DEBUG: - if previous_result is not None: - pr = previous_result.response.speech['plain'].get('original_speech', previous_result.response.speech['plain']['speech']) - result.response.speech['plain']['speech'] = f"{previous_result.response.speech['plain'].get('agent_name', 'UNKNOWN')} failed with response: {pr} Then {agent_name} responded with {r}" - else: - result.response.speech['plain']['speech'] = f"{agent_name} responded with: {r}" - - return result - - def _convert_agent_info_to_dict(self, agents_info: list[conversation.AgentInfo]) -> dict[str, str]: - """Takes a list of AgentInfo and makes it a dict of ID -> Name.""" - - agent_manager = conversation.get_agent_manager(self.hass) - - r = {} - for agent_info in agents_info: - agent = agent_manager.async_get_agent(agent_info.id) - agent_id = agent_info.id - if hasattr(agent, "registry_entry"): - agent_id = agent.registry_entry.entity_id - r[agent_id] = agent_info.name - _LOGGER.debug("agent_id %s has name %s", agent_id, agent_info.name) - return r + _LOGGER.debug("Migration to version %s successful", config_entry.version) + + return True diff --git a/custom_components/fallback_conversation/config_flow.py b/custom_components/fallback_conversation/config_flow.py index 82ee43e..d8a3977 100644 --- a/custom_components/fallback_conversation/config_flow.py +++ b/custom_components/fallback_conversation/config_flow.py @@ -9,10 +9,10 @@ from homeassistant import config_entries from homeassistant.const import CONF_NAME -from homeassistant.core import HomeAssistant, async_get_hass, callback +from homeassistant.core import callback from homeassistant.data_entry_flow import FlowResult from homeassistant.helpers.selector import ( - ConversationAgentSelector, + ConversationAgentSelector, ConversationAgentSelectorConfig, SelectSelector, SelectSelectorConfig, @@ -53,19 +53,19 @@ class ConfigFlow(config_entries.ConfigFlow, domain=DOMAIN): """Fallback Agent config flow.""" - VERSION = 1 + VERSION = 2 async def async_step_user(self, user_input: dict[str, Any] | None = None) -> FlowResult: """Handle the initial step.""" _LOGGER.debug("ConfigFlow::user_input %s", user_input) if user_input is None: return self.async_show_form( - step_id="user", - data_schema=STEP_USER_DATA_SCHEMA, + step_id="user", + data_schema=STEP_USER_DATA_SCHEMA, ) return self.async_create_entry( - title=user_input.get(CONF_NAME, DEFAULT_NAME), + title=user_input.get(CONF_NAME, DEFAULT_NAME), data=user_input, ) @@ -91,7 +91,7 @@ async def async_step_init( if user_input is not None: self._options.update(user_input) return self.async_create_entry( - title=user_input.get(CONF_NAME, DEFAULT_NAME), + title=user_input.get(CONF_NAME, DEFAULT_NAME), data=self._options, ) @@ -106,7 +106,7 @@ async def fallback_config_option_schema(self, options: dict) -> dict: """Return a schema for Fallback options.""" return { vol.Required( - CONF_DEBUG_LEVEL, + CONF_DEBUG_LEVEL, description={"suggested_value": options.get(CONF_DEBUG_LEVEL, DEFAULT_DEBUG_LEVEL)}, default=DEFAULT_DEBUG_LEVEL, ): SelectSelector( diff --git a/custom_components/fallback_conversation/conversation.py b/custom_components/fallback_conversation/conversation.py new file mode 100644 index 0000000..319dfa9 --- /dev/null +++ b/custom_components/fallback_conversation/conversation.py @@ -0,0 +1,196 @@ +"""Fallback Conversation Agent""" +from __future__ import annotations + +import logging + +from homeassistant.components import assist_pipeline, conversation +from homeassistant.config_entries import ConfigEntry +from homeassistant.core import HomeAssistant +from homeassistant.util import ulid +from homeassistant.helpers.entity_platform import AddEntitiesCallback +from homeassistant.const import MATCH_ALL + +from homeassistant.helpers import ( + config_validation as cv, + intent, +) + +from .const import ( + CONF_DEBUG_LEVEL, + CONF_PRIMARY_AGENT, + CONF_FALLBACK_AGENT, + DEBUG_LEVEL_NO_DEBUG, + DEBUG_LEVEL_LOW_DEBUG, + DEBUG_LEVEL_VERBOSE_DEBUG, + DOMAIN, + STRANGE_ERROR_RESPONSES, +) + +_LOGGER = logging.getLogger(__name__) + +CONFIG_SCHEMA = cv.config_entry_only_config_schema(DOMAIN) + + +async def async_setup_entry(hass: HomeAssistant, entry: ConfigEntry, async_add_entities: AddEntitiesCallback) -> bool: + """Set up Fallback Conversation from a config entry.""" + agent = FallbackConversationAgent(hass, entry) + async_add_entities([agent]) + return True + +class FallbackConversationAgent(conversation.ConversationEntity, conversation.AbstractConversationAgent): + """Fallback Conversation Agent.""" + + last_used_agent: str | None + + entry: ConfigEntry + hass: HomeAssistant + + _attr_has_entity_name = True + + def __init__(self, hass: HomeAssistant, entry: ConfigEntry) -> None: + """Initialize the agent.""" + self.hass = hass + self.entry = entry + self.last_used_agent = None + self._attr_name = entry.title + self._attr_unique_id = entry.entry_id + self._attr_supported_features = ( + conversation.ConversationEntityFeature.CONTROL + ) + self.in_context_examples = None + + @property + def supported_languages(self) -> list[str]: + """Return a list of supported languages.""" + return MATCH_ALL + + async def async_added_to_hass(self) -> None: + """When entity is added to Home Assistant.""" + await super().async_added_to_hass() + assist_pipeline.async_migrate_engine( + self.hass, "conversation", self.entry.entry_id, self.entity_id + ) + conversation.async_set_agent(self.hass, self.entry, self) + self.entry.async_on_unload( + self.entry.add_update_listener(self._async_entry_update_listener) + ) + + async def async_will_remove_from_hass(self) -> None: + """When entity will be removed from Home Assistant.""" + conversation.async_unset_agent(self.hass, self.entry) + await super().async_will_remove_from_hass() + + async def _async_entry_update_listener( + self, hass: HomeAssistant, entry: ConfigEntry + ) -> None: + """Handle options update.""" + self._attr_supported_features = ( + conversation.ConversationEntityFeature.CONTROL + ) + + async def async_process( + self, user_input: conversation.ConversationInput + ) -> conversation.ConversationResult: + """Process a sentence.""" + agent_manager = conversation.get_agent_manager(self.hass) + default_agent = conversation.default_agent.async_get_default_agent(self.hass) + agent_names = self._convert_agent_info_to_dict( + agent_manager.async_get_agent_info() + ) + agent_names[conversation.const.HOME_ASSISTANT_AGENT] = default_agent.name + agent_names[conversation.const.OLD_HOME_ASSISTANT_AGENT] = default_agent.name + agents = [ + self.entry.options.get(CONF_PRIMARY_AGENT, default_agent), + self.entry.options.get(CONF_FALLBACK_AGENT, default_agent), + ] + + debug_level = self.entry.options.get(CONF_DEBUG_LEVEL, DEBUG_LEVEL_NO_DEBUG) + + if user_input.conversation_id is None: + user_input.conversation_id = ulid.ulid() + + all_results = [] + result = None + for agent_id in agents: + agent_name = "[unknown]" + if agent_id in agent_names: + agent_name = agent_names[agent_id] + else: + _LOGGER.warning("agent_name not found for agent_id %s", agent_id) + + result = await self._async_process_agent( + agent_manager, + agent_id, + agent_name, + user_input, + debug_level, + result, + ) + if result.response.response_type != intent.IntentResponseType.ERROR and result.response.speech['plain']['original_speech'].lower() not in STRANGE_ERROR_RESPONSES: + return result + all_results.append(result) + + intent_response = intent.IntentResponse(language=user_input.language) + err = "Complete fallback failure. No Conversation Agent was able to respond." + if debug_level == DEBUG_LEVEL_LOW_DEBUG: + r = all_results[-1].response.speech['plain'] + err += f"\n{r.get('agent_name', 'UNKNOWN')} responded with: {r.get('original_speech', r['speech'])}" + elif debug_level == DEBUG_LEVEL_VERBOSE_DEBUG: + for res in all_results: + r = res.response.speech['plain'] + err += f"\n{r.get('agent_name', 'UNKNOWN')} responded with: {r.get('original_speech', r['speech'])}" + intent_response.async_set_error( + intent.IntentResponseErrorCode.NO_INTENT_MATCH, + err, + ) + result = conversation.ConversationResult( + conversation_id=result.conversation_id, + response=intent_response + ) + + return result + + async def _async_process_agent( + self, + agent_manager: conversation.AgentManager, + agent_id: str, + agent_name: str, + user_input: conversation.ConversationInput, + debug_level: int, + previous_result, + ) -> conversation.ConversationResult: + """Process a specified agent.""" + agent = conversation.agent_manager.async_get_agent(self.hass, agent_id) + + _LOGGER.debug("Processing in %s using %s with debug level %s: %s", user_input.language, agent_id, debug_level, user_input.text) + + result = await agent.async_process(user_input) + r = result.response.speech['plain']['speech'] + result.response.speech['plain']['original_speech'] = r + result.response.speech['plain']['agent_name'] = agent_name + result.response.speech['plain']['agent_id'] = agent_id + if debug_level == DEBUG_LEVEL_LOW_DEBUG: + result.response.speech['plain']['speech'] = f"{agent_name} responded with: {r}" + elif debug_level == DEBUG_LEVEL_VERBOSE_DEBUG: + if previous_result is not None: + pr = previous_result.response.speech['plain'].get('original_speech', previous_result.response.speech['plain']['speech']) + result.response.speech['plain']['speech'] = f"{previous_result.response.speech['plain'].get('agent_name', 'UNKNOWN')} failed with response: {pr} Then {agent_name} responded with {r}" + else: + result.response.speech['plain']['speech'] = f"{agent_name} responded with: {r}" + + return result + + def _convert_agent_info_to_dict(self, agents_info: list[conversation.AgentInfo]) -> dict[str, str]: + """Takes a list of AgentInfo and makes it a dict of ID -> Name.""" + + agent_manager = conversation.get_agent_manager(self.hass) + + r = {} + for agent_info in agents_info: + agent = agent_manager.async_get_agent(agent_info.id) + agent_id = agent_info.id + if hasattr(agent, "registry_entry"): + agent_id = agent.registry_entry.entity_id + r[agent_id] = agent_info.name + _LOGGER.debug("agent_id %s has name %s", agent_id, agent_info.name) + return r \ No newline at end of file diff --git a/custom_components/fallback_conversation/manifest.json b/custom_components/fallback_conversation/manifest.json index 89f27da..415acb5 100644 --- a/custom_components/fallback_conversation/manifest.json +++ b/custom_components/fallback_conversation/manifest.json @@ -8,10 +8,11 @@ "dependencies": [ "conversation" ], + "after_dependencies": ["assist_pipeline"], "documentation": "https://github.com/m50/ha-fallback-conversation", "integration_type": "service", "iot_class": "local_polling", "issue_tracker": "https://github.com/m50/ha-fallback-conversation/issues", "requirements": [], - "version": "1.0.5" + "version": "1.1.0" } \ No newline at end of file