2019-02-13 20:21:14 +00:00
|
|
|
"""Support for functionality to have conversations with Home Assistant."""
|
2015-03-10 07:08:50 +00:00
|
|
|
import logging
|
|
|
|
import re
|
|
|
|
|
2016-04-13 16:48:39 +00:00
|
|
|
import voluptuous as vol
|
|
|
|
|
2015-08-17 03:44:46 +00:00
|
|
|
from homeassistant import core
|
2019-11-08 17:06:23 +00:00
|
|
|
from homeassistant.components import http, websocket_api
|
2019-03-21 05:56:46 +00:00
|
|
|
from homeassistant.components.http.data_validator import RequestDataValidator
|
2020-04-08 21:20:03 +00:00
|
|
|
from homeassistant.const import HTTP_INTERNAL_SERVER_ERROR
|
2019-03-21 05:56:46 +00:00
|
|
|
from homeassistant.helpers import config_validation as cv, intent
|
2018-02-11 17:33:19 +00:00
|
|
|
from homeassistant.loader import bind_hass
|
2019-03-21 05:56:46 +00:00
|
|
|
|
2019-10-18 18:46:45 +00:00
|
|
|
from .agent import AbstractConversationAgent
|
2019-12-09 17:56:21 +00:00
|
|
|
from .default_agent import DefaultAgent, async_register
|
2015-03-10 07:08:50 +00:00
|
|
|
|
2017-12-29 09:06:39 +00:00
|
|
|
_LOGGER = logging.getLogger(__name__)
|
2016-08-22 12:19:19 +00:00
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
ATTR_TEXT = "text"
|
2017-12-29 09:06:39 +00:00
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
DOMAIN = "conversation"
|
2016-08-22 12:19:19 +00:00
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
REGEX_TYPE = type(re.compile(""))
|
2019-10-18 18:46:45 +00:00
|
|
|
DATA_AGENT = "conversation_agent"
|
2019-11-08 17:06:23 +00:00
|
|
|
DATA_CONFIG = "conversation_config"
|
2018-03-31 00:22:48 +00:00
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
SERVICE_PROCESS = "process"
|
2015-03-10 07:08:50 +00:00
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
SERVICE_PROCESS_SCHEMA = vol.Schema({vol.Required(ATTR_TEXT): cv.string})
|
2016-04-13 16:48:39 +00:00
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
CONFIG_SCHEMA = vol.Schema(
|
|
|
|
{
|
|
|
|
DOMAIN: vol.Schema(
|
|
|
|
{
|
|
|
|
vol.Optional("intents"): vol.Schema(
|
|
|
|
{cv.string: vol.All(cv.ensure_list, [cv.string])}
|
|
|
|
)
|
|
|
|
}
|
|
|
|
)
|
|
|
|
},
|
|
|
|
extra=vol.ALLOW_EXTRA,
|
|
|
|
)
|
2016-09-30 02:02:22 +00:00
|
|
|
|
2019-10-18 18:46:45 +00:00
|
|
|
async_register = bind_hass(async_register) # pylint: disable=invalid-name
|
2017-07-22 04:38:53 +00:00
|
|
|
|
|
|
|
|
2019-10-18 18:46:45 +00:00
|
|
|
@core.callback
|
|
|
|
@bind_hass
|
|
|
|
def async_set_agent(hass: core.HomeAssistant, agent: AbstractConversationAgent):
|
|
|
|
"""Set the agent to handle the conversations."""
|
|
|
|
hass.data[DATA_AGENT] = agent
|
2017-07-22 04:38:53 +00:00
|
|
|
|
|
|
|
|
2019-11-08 17:06:23 +00:00
|
|
|
async def async_setup(hass, config):
|
|
|
|
"""Register the process service."""
|
|
|
|
hass.data[DATA_CONFIG] = config
|
2017-07-22 04:38:53 +00:00
|
|
|
|
2019-10-18 18:46:45 +00:00
|
|
|
async def handle_service(service):
|
2016-03-08 16:55:57 +00:00
|
|
|
"""Parse text into commands."""
|
2016-04-13 16:48:39 +00:00
|
|
|
text = service.data[ATTR_TEXT]
|
2019-07-31 19:25:30 +00:00
|
|
|
_LOGGER.debug("Processing: <%s>", text)
|
2019-11-26 10:30:21 +00:00
|
|
|
agent = await _get_agent(hass)
|
2018-03-01 15:35:12 +00:00
|
|
|
try:
|
2019-11-26 10:30:21 +00:00
|
|
|
await agent.async_process(text, service.context)
|
2018-03-01 15:35:12 +00:00
|
|
|
except intent.IntentHandleError as err:
|
2019-07-31 19:25:30 +00:00
|
|
|
_LOGGER.error("Error processing %s: %s", text, err)
|
2017-07-22 04:38:53 +00:00
|
|
|
|
|
|
|
hass.services.async_register(
|
2019-10-18 18:46:45 +00:00
|
|
|
DOMAIN, SERVICE_PROCESS, handle_service, schema=SERVICE_PROCESS_SCHEMA
|
2019-07-31 19:25:30 +00:00
|
|
|
)
|
2019-11-08 17:06:23 +00:00
|
|
|
hass.http.register_view(ConversationProcessView())
|
|
|
|
hass.components.websocket_api.async_register_command(websocket_process)
|
|
|
|
hass.components.websocket_api.async_register_command(websocket_get_agent_info)
|
|
|
|
hass.components.websocket_api.async_register_command(websocket_set_onboarding)
|
2018-03-31 00:22:48 +00:00
|
|
|
|
2017-07-22 04:38:53 +00:00
|
|
|
return True
|
2015-03-10 07:08:50 +00:00
|
|
|
|
|
|
|
|
2019-11-08 17:06:23 +00:00
|
|
|
@websocket_api.async_response
|
|
|
|
@websocket_api.websocket_command(
|
|
|
|
{"type": "conversation/process", "text": str, vol.Optional("conversation_id"): str}
|
|
|
|
)
|
|
|
|
async def websocket_process(hass, connection, msg):
|
|
|
|
"""Process text."""
|
|
|
|
connection.send_result(
|
2019-11-26 10:30:21 +00:00
|
|
|
msg["id"],
|
|
|
|
await _async_converse(
|
|
|
|
hass, msg["text"], msg.get("conversation_id"), connection.context(msg)
|
|
|
|
),
|
2019-11-08 17:06:23 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
|
|
|
|
@websocket_api.async_response
|
|
|
|
@websocket_api.websocket_command({"type": "conversation/agent/info"})
|
|
|
|
async def websocket_get_agent_info(hass, connection, msg):
|
|
|
|
"""Do we need onboarding."""
|
2019-11-26 10:30:21 +00:00
|
|
|
agent = await _get_agent(hass)
|
2019-11-08 17:06:23 +00:00
|
|
|
|
|
|
|
connection.send_result(
|
|
|
|
msg["id"],
|
|
|
|
{
|
|
|
|
"onboarding": await agent.async_get_onboarding(),
|
|
|
|
"attribution": agent.attribution,
|
|
|
|
},
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
|
|
@websocket_api.async_response
|
|
|
|
@websocket_api.websocket_command({"type": "conversation/onboarding/set", "shown": bool})
|
|
|
|
async def websocket_set_onboarding(hass, connection, msg):
|
|
|
|
"""Set onboarding status."""
|
2019-11-26 10:30:21 +00:00
|
|
|
agent = await _get_agent(hass)
|
2019-11-08 17:06:23 +00:00
|
|
|
|
|
|
|
success = await agent.async_set_onboarding(msg["shown"])
|
|
|
|
|
|
|
|
if success:
|
|
|
|
connection.send_result(msg["id"])
|
|
|
|
else:
|
|
|
|
connection.send_error(msg["id"])
|
|
|
|
|
|
|
|
|
2017-07-22 04:38:53 +00:00
|
|
|
class ConversationProcessView(http.HomeAssistantView):
|
2019-11-08 17:06:23 +00:00
|
|
|
"""View to process text."""
|
2017-07-22 04:38:53 +00:00
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
url = "/api/conversation/process"
|
2017-07-22 04:38:53 +00:00
|
|
|
name = "api:conversation:process"
|
|
|
|
|
2019-11-07 20:21:12 +00:00
|
|
|
@RequestDataValidator(
|
|
|
|
vol.Schema({vol.Required("text"): str, vol.Optional("conversation_id"): str})
|
|
|
|
)
|
2018-03-01 15:35:12 +00:00
|
|
|
async def post(self, request, data):
|
2017-07-22 04:38:53 +00:00
|
|
|
"""Send a request for processing."""
|
2019-07-31 19:25:30 +00:00
|
|
|
hass = request.app["hass"]
|
2019-11-26 10:30:21 +00:00
|
|
|
|
2020-02-08 12:10:59 +00:00
|
|
|
try:
|
|
|
|
intent_result = await _async_converse(
|
|
|
|
hass, data["text"], data.get("conversation_id"), self.context(request)
|
|
|
|
)
|
|
|
|
except intent.IntentError as err:
|
|
|
|
_LOGGER.error("Error handling intent: %s", err)
|
|
|
|
return self.json(
|
|
|
|
{
|
|
|
|
"success": False,
|
|
|
|
"error": {
|
|
|
|
"code": str(err.__class__.__name__).lower(),
|
|
|
|
"message": str(err),
|
|
|
|
},
|
|
|
|
},
|
2020-04-08 21:20:03 +00:00
|
|
|
status_code=HTTP_INTERNAL_SERVER_ERROR,
|
2020-02-08 12:10:59 +00:00
|
|
|
)
|
2017-07-25 07:42:59 +00:00
|
|
|
|
2017-07-22 04:38:53 +00:00
|
|
|
return self.json(intent_result)
|
2019-11-19 19:47:03 +00:00
|
|
|
|
|
|
|
|
2019-11-26 10:30:21 +00:00
|
|
|
async def _get_agent(hass: core.HomeAssistant) -> AbstractConversationAgent:
|
|
|
|
"""Get the active conversation agent."""
|
|
|
|
agent = hass.data.get(DATA_AGENT)
|
|
|
|
if agent is None:
|
|
|
|
agent = hass.data[DATA_AGENT] = DefaultAgent(hass)
|
|
|
|
await agent.async_initialize(hass.data.get(DATA_CONFIG))
|
|
|
|
return agent
|
|
|
|
|
|
|
|
|
|
|
|
async def _async_converse(
|
|
|
|
hass: core.HomeAssistant, text: str, conversation_id: str, context: core.Context
|
|
|
|
) -> intent.IntentResponse:
|
|
|
|
"""Process text and get intent."""
|
|
|
|
agent = await _get_agent(hass)
|
|
|
|
try:
|
|
|
|
intent_result = await agent.async_process(text, context, conversation_id)
|
|
|
|
except intent.IntentHandleError as err:
|
|
|
|
intent_result = intent.IntentResponse()
|
|
|
|
intent_result.async_set_speech(str(err))
|
|
|
|
|
|
|
|
if intent_result is None:
|
|
|
|
intent_result = intent.IntentResponse()
|
|
|
|
intent_result.async_set_speech("Sorry, I didn't understand that")
|
|
|
|
|
|
|
|
return intent_result
|