2016-03-09 09:25:50 +00:00
|
|
|
"""The tests for the Conversation component."""
|
2021-10-22 17:43:40 +00:00
|
|
|
from http import HTTPStatus
|
2022-12-23 01:19:37 +00:00
|
|
|
from unittest.mock import ANY, patch
|
2021-10-22 17:43:40 +00:00
|
|
|
|
2017-11-21 04:26:36 +00:00
|
|
|
import pytest
|
|
|
|
|
2015-09-01 07:18:26 +00:00
|
|
|
from homeassistant.components import conversation
|
2023-01-07 21:20:21 +00:00
|
|
|
from homeassistant.core import DOMAIN as HASS_DOMAIN
|
2017-07-22 04:38:53 +00:00
|
|
|
from homeassistant.helpers import intent
|
2019-12-09 17:56:21 +00:00
|
|
|
from homeassistant.setup import async_setup_component
|
2015-08-30 08:24:24 +00:00
|
|
|
|
2023-01-07 21:20:21 +00:00
|
|
|
from tests.common import async_mock_service
|
2017-06-13 06:34:20 +00:00
|
|
|
|
|
|
|
|
2023-01-09 22:48:59 +00:00
|
|
|
class OrderBeerIntentHandler(intent.IntentHandler):
|
|
|
|
"""Handle OrderBeer intent."""
|
|
|
|
|
|
|
|
intent_type = "OrderBeer"
|
|
|
|
|
|
|
|
async def async_handle(self, intent_obj: intent.Intent) -> intent.IntentResponse:
|
|
|
|
"""Return speech response."""
|
|
|
|
beer_style = intent_obj.slots["beer_style"]["value"]
|
|
|
|
response = intent_obj.create_response()
|
|
|
|
response.async_set_speech(f"You ordered a {beer_style}")
|
|
|
|
return response
|
|
|
|
|
|
|
|
|
2022-12-09 01:30:08 +00:00
|
|
|
@pytest.fixture
|
|
|
|
async def init_components(hass):
|
|
|
|
"""Initialize relevant components with empty configs."""
|
|
|
|
assert await async_setup_component(hass, "homeassistant", {})
|
|
|
|
assert await async_setup_component(hass, "conversation", {})
|
|
|
|
assert await async_setup_component(hass, "intent", {})
|
|
|
|
|
|
|
|
|
2023-01-07 21:20:21 +00:00
|
|
|
async def test_http_processing_intent(
|
|
|
|
hass, init_components, hass_client, hass_admin_user
|
|
|
|
):
|
2017-07-22 04:38:53 +00:00
|
|
|
"""Test processing intent via HTTP API."""
|
2023-01-07 21:20:21 +00:00
|
|
|
hass.states.async_set("light.kitchen", "on")
|
2018-11-27 09:41:44 +00:00
|
|
|
client = await hass_client()
|
2019-07-31 19:25:30 +00:00
|
|
|
resp = await client.post(
|
2023-01-07 21:20:21 +00:00
|
|
|
"/api/conversation/process", json={"text": "turn on kitchen"}
|
2019-07-31 19:25:30 +00:00
|
|
|
)
|
2017-07-22 04:38:53 +00:00
|
|
|
|
2021-10-22 17:43:40 +00:00
|
|
|
assert resp.status == HTTPStatus.OK
|
2018-03-31 00:22:48 +00:00
|
|
|
data = await resp.json()
|
2017-07-22 04:38:53 +00:00
|
|
|
|
|
|
|
assert data == {
|
2022-12-13 22:46:40 +00:00
|
|
|
"response": {
|
|
|
|
"response_type": "action_done",
|
2023-01-07 21:20:21 +00:00
|
|
|
"card": {},
|
2022-12-13 22:46:40 +00:00
|
|
|
"speech": {
|
|
|
|
"plain": {
|
|
|
|
"extra_data": None,
|
2023-01-07 21:20:21 +00:00
|
|
|
"speech": "Turned kitchen on",
|
2022-12-13 22:46:40 +00:00
|
|
|
}
|
|
|
|
},
|
|
|
|
"language": hass.config.language,
|
|
|
|
"data": {
|
2023-01-07 21:20:21 +00:00
|
|
|
"targets": [],
|
|
|
|
"success": [
|
|
|
|
{"id": "light.kitchen", "name": "kitchen", "type": "entity"}
|
|
|
|
],
|
|
|
|
"failed": [],
|
2022-12-13 22:46:40 +00:00
|
|
|
},
|
|
|
|
},
|
|
|
|
"conversation_id": None,
|
2017-07-22 04:38:53 +00:00
|
|
|
}
|
2017-11-21 04:26:36 +00:00
|
|
|
|
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
@pytest.mark.parametrize("sentence", ("turn on kitchen", "turn kitchen on"))
|
2022-12-09 01:30:08 +00:00
|
|
|
async def test_turn_on_intent(hass, init_components, sentence):
|
2017-11-21 04:26:36 +00:00
|
|
|
"""Test calling the turn on intent."""
|
2019-07-31 19:25:30 +00:00
|
|
|
hass.states.async_set("light.kitchen", "off")
|
|
|
|
calls = async_mock_service(hass, HASS_DOMAIN, "turn_on")
|
2017-11-21 04:26:36 +00:00
|
|
|
|
2018-03-31 00:22:48 +00:00
|
|
|
await hass.services.async_call(
|
2019-07-31 19:25:30 +00:00
|
|
|
"conversation", "process", {conversation.ATTR_TEXT: sentence}
|
|
|
|
)
|
2018-03-31 00:22:48 +00:00
|
|
|
await hass.async_block_till_done()
|
2017-11-21 04:26:36 +00:00
|
|
|
|
|
|
|
assert len(calls) == 1
|
|
|
|
call = calls[0]
|
2018-10-30 15:38:09 +00:00
|
|
|
assert call.domain == HASS_DOMAIN
|
2019-07-31 19:25:30 +00:00
|
|
|
assert call.service == "turn_on"
|
|
|
|
assert call.data == {"entity_id": "light.kitchen"}
|
2017-11-21 04:26:36 +00:00
|
|
|
|
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
@pytest.mark.parametrize("sentence", ("turn off kitchen", "turn kitchen off"))
|
2022-12-09 01:30:08 +00:00
|
|
|
async def test_turn_off_intent(hass, init_components, sentence):
|
2017-11-21 04:26:36 +00:00
|
|
|
"""Test calling the turn on intent."""
|
2019-07-31 19:25:30 +00:00
|
|
|
hass.states.async_set("light.kitchen", "on")
|
|
|
|
calls = async_mock_service(hass, HASS_DOMAIN, "turn_off")
|
2017-11-21 04:26:36 +00:00
|
|
|
|
2018-03-31 00:22:48 +00:00
|
|
|
await hass.services.async_call(
|
2019-07-31 19:25:30 +00:00
|
|
|
"conversation", "process", {conversation.ATTR_TEXT: sentence}
|
|
|
|
)
|
2018-03-31 00:22:48 +00:00
|
|
|
await hass.async_block_till_done()
|
2017-11-21 04:26:36 +00:00
|
|
|
|
|
|
|
assert len(calls) == 1
|
|
|
|
call = calls[0]
|
2018-10-30 15:38:09 +00:00
|
|
|
assert call.domain == HASS_DOMAIN
|
2019-07-31 19:25:30 +00:00
|
|
|
assert call.service == "turn_off"
|
|
|
|
assert call.data == {"entity_id": "light.kitchen"}
|
2017-11-21 04:26:36 +00:00
|
|
|
|
|
|
|
|
2022-12-09 01:30:08 +00:00
|
|
|
async def test_http_api(hass, init_components, hass_client):
|
2017-11-21 04:26:36 +00:00
|
|
|
"""Test the HTTP conversation API."""
|
2018-11-27 09:41:44 +00:00
|
|
|
client = await hass_client()
|
2019-07-31 19:25:30 +00:00
|
|
|
hass.states.async_set("light.kitchen", "off")
|
|
|
|
calls = async_mock_service(hass, HASS_DOMAIN, "turn_on")
|
2017-11-21 04:26:36 +00:00
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
resp = await client.post(
|
|
|
|
"/api/conversation/process", json={"text": "Turn the kitchen on"}
|
|
|
|
)
|
2021-10-22 17:43:40 +00:00
|
|
|
assert resp.status == HTTPStatus.OK
|
2022-12-09 01:30:08 +00:00
|
|
|
data = await resp.json()
|
|
|
|
|
|
|
|
assert data == {
|
2022-12-13 22:46:40 +00:00
|
|
|
"response": {
|
|
|
|
"card": {},
|
|
|
|
"speech": {"plain": {"extra_data": None, "speech": "Turned kitchen on"}},
|
|
|
|
"language": hass.config.language,
|
|
|
|
"response_type": "action_done",
|
|
|
|
"data": {
|
2022-12-14 04:32:30 +00:00
|
|
|
"targets": [],
|
|
|
|
"success": [
|
2022-12-13 22:46:40 +00:00
|
|
|
{
|
|
|
|
"type": "entity",
|
|
|
|
"name": "kitchen",
|
|
|
|
"id": "light.kitchen",
|
|
|
|
},
|
2022-12-14 04:32:30 +00:00
|
|
|
],
|
|
|
|
"failed": [],
|
2022-12-13 22:46:40 +00:00
|
|
|
},
|
2022-12-09 01:30:08 +00:00
|
|
|
},
|
2022-12-13 22:46:40 +00:00
|
|
|
"conversation_id": None,
|
2022-12-09 01:30:08 +00:00
|
|
|
}
|
2017-11-21 04:26:36 +00:00
|
|
|
|
|
|
|
assert len(calls) == 1
|
|
|
|
call = calls[0]
|
2018-10-30 15:38:09 +00:00
|
|
|
assert call.domain == HASS_DOMAIN
|
2019-07-31 19:25:30 +00:00
|
|
|
assert call.service == "turn_on"
|
|
|
|
assert call.data == {"entity_id": "light.kitchen"}
|
2017-11-21 04:26:36 +00:00
|
|
|
|
|
|
|
|
2022-12-09 01:30:08 +00:00
|
|
|
async def test_http_api_no_match(hass, init_components, hass_client):
|
|
|
|
"""Test the HTTP conversation API with an intent match failure."""
|
|
|
|
client = await hass_client()
|
|
|
|
|
2023-01-07 21:20:21 +00:00
|
|
|
# Shouldn't match any intents
|
2022-12-09 01:30:08 +00:00
|
|
|
resp = await client.post("/api/conversation/process", json={"text": "do something"})
|
|
|
|
|
|
|
|
assert resp.status == HTTPStatus.OK
|
|
|
|
data = await resp.json()
|
|
|
|
|
|
|
|
assert data == {
|
2022-12-13 22:46:40 +00:00
|
|
|
"response": {
|
|
|
|
"response_type": "error",
|
|
|
|
"card": {},
|
|
|
|
"speech": {
|
|
|
|
"plain": {
|
2023-01-07 21:20:21 +00:00
|
|
|
"speech": "Sorry, I didn't understand that",
|
2022-12-13 22:46:40 +00:00
|
|
|
"extra_data": None,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
"language": hass.config.language,
|
2023-01-07 21:20:21 +00:00
|
|
|
"data": {"code": "no_intent_match"},
|
2022-12-09 01:30:08 +00:00
|
|
|
},
|
2022-12-13 22:46:40 +00:00
|
|
|
"conversation_id": None,
|
2022-12-09 01:30:08 +00:00
|
|
|
}
|
2018-02-11 17:33:19 +00:00
|
|
|
|
2017-11-21 04:26:36 +00:00
|
|
|
|
2022-12-09 01:30:08 +00:00
|
|
|
async def test_http_api_handle_failure(hass, init_components, hass_client):
|
|
|
|
"""Test the HTTP conversation API with an error during handling."""
|
|
|
|
client = await hass_client()
|
|
|
|
|
|
|
|
hass.states.async_set("light.kitchen", "off")
|
|
|
|
|
|
|
|
# Raise an "unexpected" error during intent handling
|
|
|
|
def async_handle_error(*args, **kwargs):
|
|
|
|
raise intent.IntentUnexpectedError(
|
|
|
|
"Unexpected error turning on the kitchen light"
|
|
|
|
)
|
|
|
|
|
|
|
|
with patch("homeassistant.helpers.intent.async_handle", new=async_handle_error):
|
|
|
|
resp = await client.post(
|
|
|
|
"/api/conversation/process", json={"text": "turn on the kitchen"}
|
|
|
|
)
|
|
|
|
|
|
|
|
assert resp.status == HTTPStatus.OK
|
|
|
|
data = await resp.json()
|
|
|
|
|
|
|
|
assert data == {
|
2022-12-13 22:46:40 +00:00
|
|
|
"response": {
|
|
|
|
"response_type": "error",
|
|
|
|
"card": {},
|
|
|
|
"speech": {
|
|
|
|
"plain": {
|
|
|
|
"extra_data": None,
|
|
|
|
"speech": "Unexpected error turning on the kitchen light",
|
|
|
|
}
|
|
|
|
},
|
|
|
|
"language": hass.config.language,
|
|
|
|
"data": {
|
|
|
|
"code": "failed_to_handle",
|
|
|
|
},
|
2022-12-09 01:30:08 +00:00
|
|
|
},
|
2022-12-13 22:46:40 +00:00
|
|
|
"conversation_id": None,
|
2022-12-09 01:30:08 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
async def test_http_api_wrong_data(hass, init_components, hass_client):
|
|
|
|
"""Test the HTTP conversation API."""
|
2018-11-27 09:41:44 +00:00
|
|
|
client = await hass_client()
|
2017-11-21 04:26:36 +00:00
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
resp = await client.post("/api/conversation/process", json={"text": 123})
|
2021-10-22 17:43:40 +00:00
|
|
|
assert resp.status == HTTPStatus.BAD_REQUEST
|
2017-11-21 04:26:36 +00:00
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
resp = await client.post("/api/conversation/process", json={})
|
2021-10-22 17:43:40 +00:00
|
|
|
assert resp.status == HTTPStatus.BAD_REQUEST
|
2018-03-01 15:35:12 +00:00
|
|
|
|
|
|
|
|
2019-11-26 10:30:21 +00:00
|
|
|
async def test_custom_agent(hass, hass_client, hass_admin_user):
|
2019-10-18 18:46:45 +00:00
|
|
|
"""Test a custom conversation agent."""
|
|
|
|
|
2019-11-07 20:21:12 +00:00
|
|
|
calls = []
|
|
|
|
|
2019-10-18 18:46:45 +00:00
|
|
|
class MyAgent(conversation.AbstractConversationAgent):
|
|
|
|
"""Test Agent."""
|
|
|
|
|
2022-12-08 16:39:28 +00:00
|
|
|
async def async_process(self, text, context, conversation_id, language):
|
2019-10-18 18:46:45 +00:00
|
|
|
"""Process some text."""
|
2022-12-08 16:39:28 +00:00
|
|
|
calls.append((text, context, conversation_id, language))
|
|
|
|
response = intent.IntentResponse(language=language)
|
2019-10-18 18:46:45 +00:00
|
|
|
response.async_set_speech("Test response")
|
2022-12-13 22:46:40 +00:00
|
|
|
return conversation.ConversationResult(
|
|
|
|
response=response, conversation_id=conversation_id
|
|
|
|
)
|
2019-10-18 18:46:45 +00:00
|
|
|
|
|
|
|
conversation.async_set_agent(hass, MyAgent())
|
|
|
|
|
|
|
|
assert await async_setup_component(hass, "conversation", {})
|
|
|
|
|
|
|
|
client = await hass_client()
|
|
|
|
|
2019-11-07 20:21:12 +00:00
|
|
|
resp = await client.post(
|
|
|
|
"/api/conversation/process",
|
2022-12-08 16:39:28 +00:00
|
|
|
json={
|
|
|
|
"text": "Test Text",
|
|
|
|
"conversation_id": "test-conv-id",
|
|
|
|
"language": "test-language",
|
|
|
|
},
|
2019-11-07 20:21:12 +00:00
|
|
|
)
|
2021-10-22 17:43:40 +00:00
|
|
|
assert resp.status == HTTPStatus.OK
|
2019-10-18 18:46:45 +00:00
|
|
|
assert await resp.json() == {
|
2022-12-13 22:46:40 +00:00
|
|
|
"response": {
|
|
|
|
"response_type": "action_done",
|
|
|
|
"card": {},
|
|
|
|
"speech": {
|
|
|
|
"plain": {
|
|
|
|
"extra_data": None,
|
|
|
|
"speech": "Test response",
|
|
|
|
}
|
|
|
|
},
|
|
|
|
"language": "test-language",
|
2022-12-14 04:32:30 +00:00
|
|
|
"data": {"targets": [], "success": [], "failed": []},
|
2022-12-08 16:39:28 +00:00
|
|
|
},
|
2022-12-13 22:46:40 +00:00
|
|
|
"conversation_id": "test-conv-id",
|
2019-10-18 18:46:45 +00:00
|
|
|
}
|
2019-11-07 20:21:12 +00:00
|
|
|
|
|
|
|
assert len(calls) == 1
|
|
|
|
assert calls[0][0] == "Test Text"
|
2019-11-26 10:30:21 +00:00
|
|
|
assert calls[0][1].user_id == hass_admin_user.id
|
|
|
|
assert calls[0][2] == "test-conv-id"
|
2022-12-08 16:39:28 +00:00
|
|
|
assert calls[0][3] == "test-language"
|
2022-12-23 01:19:37 +00:00
|
|
|
|
|
|
|
|
|
|
|
@pytest.mark.parametrize(
|
|
|
|
"payload",
|
|
|
|
[
|
|
|
|
{
|
|
|
|
"text": "Test Text",
|
|
|
|
},
|
|
|
|
{
|
|
|
|
"text": "Test Text",
|
|
|
|
"language": "test-language",
|
|
|
|
},
|
|
|
|
{
|
|
|
|
"text": "Test Text",
|
|
|
|
"conversation_id": "test-conv-id",
|
|
|
|
},
|
|
|
|
{
|
|
|
|
"text": "Test Text",
|
|
|
|
"conversation_id": None,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
"text": "Test Text",
|
|
|
|
"conversation_id": "test-conv-id",
|
|
|
|
"language": "test-language",
|
|
|
|
},
|
|
|
|
],
|
|
|
|
)
|
|
|
|
async def test_ws_api(hass, hass_ws_client, payload):
|
|
|
|
"""Test the Websocket conversation API."""
|
|
|
|
assert await async_setup_component(hass, "conversation", {})
|
|
|
|
client = await hass_ws_client(hass)
|
|
|
|
|
|
|
|
await client.send_json({"id": 5, "type": "conversation/process", **payload})
|
|
|
|
|
|
|
|
msg = await client.receive_json()
|
|
|
|
|
|
|
|
assert msg["success"]
|
|
|
|
assert msg["result"] == {
|
|
|
|
"response": {
|
|
|
|
"response_type": "error",
|
|
|
|
"card": {},
|
|
|
|
"speech": {
|
|
|
|
"plain": {
|
|
|
|
"extra_data": None,
|
|
|
|
"speech": "Sorry, I didn't understand that",
|
|
|
|
}
|
|
|
|
},
|
|
|
|
"language": payload.get("language", hass.config.language),
|
|
|
|
"data": {"code": "no_intent_match"},
|
|
|
|
},
|
|
|
|
"conversation_id": payload.get("conversation_id") or ANY,
|
|
|
|
}
|
2023-01-09 22:48:59 +00:00
|
|
|
|
|
|
|
|
2023-01-19 01:36:51 +00:00
|
|
|
# pylint: disable=protected-access
|
|
|
|
async def test_ws_prepare(hass, hass_ws_client):
|
|
|
|
"""Test the Websocket prepare conversation API."""
|
|
|
|
assert await async_setup_component(hass, "conversation", {})
|
|
|
|
agent = await conversation._get_agent(hass)
|
|
|
|
assert isinstance(agent, conversation.DefaultAgent)
|
|
|
|
|
|
|
|
# No intents should be loaded yet
|
|
|
|
assert not agent._lang_intents.get(hass.config.language)
|
|
|
|
|
|
|
|
client = await hass_ws_client(hass)
|
|
|
|
|
|
|
|
await client.send_json(
|
|
|
|
{
|
|
|
|
"id": 5,
|
|
|
|
"type": "conversation/prepare",
|
|
|
|
}
|
|
|
|
)
|
|
|
|
|
|
|
|
msg = await client.receive_json()
|
|
|
|
|
|
|
|
assert msg["success"]
|
|
|
|
assert msg["id"] == 5
|
|
|
|
|
|
|
|
# Intents should now be load
|
|
|
|
assert agent._lang_intents.get(hass.config.language)
|
|
|
|
|
|
|
|
|
2023-01-09 22:48:59 +00:00
|
|
|
async def test_custom_sentences(hass, hass_client, hass_admin_user):
|
|
|
|
"""Test custom sentences with a custom intent."""
|
|
|
|
assert await async_setup_component(hass, "homeassistant", {})
|
|
|
|
assert await async_setup_component(hass, "conversation", {})
|
|
|
|
assert await async_setup_component(hass, "intent", {})
|
|
|
|
|
|
|
|
# Expecting testing_config/custom_sentences/en/beer.yaml
|
|
|
|
intent.async_register(hass, OrderBeerIntentHandler())
|
|
|
|
|
|
|
|
# Invoke intent via HTTP API
|
|
|
|
client = await hass_client()
|
|
|
|
for beer_style in ("stout", "lager"):
|
|
|
|
resp = await client.post(
|
|
|
|
"/api/conversation/process",
|
|
|
|
json={"text": f"I'd like to order a {beer_style}, please"},
|
|
|
|
)
|
|
|
|
assert resp.status == HTTPStatus.OK
|
|
|
|
data = await resp.json()
|
|
|
|
|
|
|
|
assert data == {
|
|
|
|
"response": {
|
|
|
|
"card": {},
|
|
|
|
"speech": {
|
|
|
|
"plain": {
|
|
|
|
"extra_data": None,
|
|
|
|
"speech": f"You ordered a {beer_style}",
|
|
|
|
}
|
|
|
|
},
|
|
|
|
"language": hass.config.language,
|
|
|
|
"response_type": "action_done",
|
|
|
|
"data": {
|
|
|
|
"targets": [],
|
|
|
|
"success": [],
|
|
|
|
"failed": [],
|
|
|
|
},
|
|
|
|
},
|
|
|
|
"conversation_id": None,
|
|
|
|
}
|
2023-01-19 01:36:51 +00:00
|
|
|
|
|
|
|
|
|
|
|
# pylint: disable=protected-access
|
|
|
|
async def test_prepare_reload(hass):
|
|
|
|
"""Test calling the reload service."""
|
|
|
|
language = hass.config.language
|
|
|
|
assert await async_setup_component(hass, "conversation", {})
|
|
|
|
|
|
|
|
# Load intents
|
|
|
|
agent = await conversation._get_agent(hass)
|
|
|
|
assert isinstance(agent, conversation.DefaultAgent)
|
|
|
|
await agent.async_prepare(language)
|
|
|
|
|
|
|
|
# Confirm intents are loaded
|
|
|
|
assert agent._lang_intents.get(language)
|
|
|
|
|
|
|
|
# Clear cache
|
|
|
|
await hass.services.async_call("conversation", "reload", {})
|
|
|
|
await hass.async_block_till_done()
|
|
|
|
|
|
|
|
# Confirm intent cache is cleared
|
|
|
|
assert not agent._lang_intents.get(language)
|
|
|
|
|
|
|
|
|
|
|
|
# pylint: disable=protected-access
|
|
|
|
async def test_prepare_fail(hass):
|
|
|
|
"""Test calling prepare with a non-existent language."""
|
|
|
|
assert await async_setup_component(hass, "conversation", {})
|
|
|
|
|
|
|
|
# Load intents
|
|
|
|
agent = await conversation._get_agent(hass)
|
|
|
|
assert isinstance(agent, conversation.DefaultAgent)
|
|
|
|
await agent.async_prepare("not-a-language")
|
|
|
|
|
|
|
|
# Confirm no intents were loaded
|
|
|
|
assert not agent._lang_intents.get("not-a-language")
|