mirror of https://github.com/home-assistant/core
240 lines
7.3 KiB
Python
240 lines
7.3 KiB
Python
"""Test the Anthropic config flow."""
|
|
|
|
from unittest.mock import AsyncMock, patch
|
|
|
|
from anthropic import (
|
|
APIConnectionError,
|
|
APIResponseValidationError,
|
|
APITimeoutError,
|
|
AuthenticationError,
|
|
BadRequestError,
|
|
InternalServerError,
|
|
)
|
|
from httpx import URL, Request, Response
|
|
import pytest
|
|
|
|
from homeassistant import config_entries
|
|
from homeassistant.components.anthropic.config_flow import RECOMMENDED_OPTIONS
|
|
from homeassistant.components.anthropic.const import (
|
|
CONF_CHAT_MODEL,
|
|
CONF_MAX_TOKENS,
|
|
CONF_PROMPT,
|
|
CONF_RECOMMENDED,
|
|
CONF_TEMPERATURE,
|
|
DOMAIN,
|
|
RECOMMENDED_CHAT_MODEL,
|
|
RECOMMENDED_MAX_TOKENS,
|
|
)
|
|
from homeassistant.const import CONF_LLM_HASS_API
|
|
from homeassistant.core import HomeAssistant
|
|
from homeassistant.data_entry_flow import FlowResultType
|
|
|
|
from tests.common import MockConfigEntry
|
|
|
|
|
|
async def test_form(hass: HomeAssistant) -> None:
|
|
"""Test we get the form."""
|
|
# Pretend we already set up a config entry.
|
|
hass.config.components.add("anthropic")
|
|
MockConfigEntry(
|
|
domain=DOMAIN,
|
|
state=config_entries.ConfigEntryState.LOADED,
|
|
).add_to_hass(hass)
|
|
|
|
result = await hass.config_entries.flow.async_init(
|
|
DOMAIN, context={"source": config_entries.SOURCE_USER}
|
|
)
|
|
assert result["type"] is FlowResultType.FORM
|
|
assert result["errors"] is None
|
|
|
|
with (
|
|
patch(
|
|
"homeassistant.components.anthropic.config_flow.anthropic.resources.messages.AsyncMessages.create",
|
|
new_callable=AsyncMock,
|
|
),
|
|
patch(
|
|
"homeassistant.components.anthropic.async_setup_entry",
|
|
return_value=True,
|
|
) as mock_setup_entry,
|
|
):
|
|
result2 = await hass.config_entries.flow.async_configure(
|
|
result["flow_id"],
|
|
{
|
|
"api_key": "bla",
|
|
},
|
|
)
|
|
await hass.async_block_till_done()
|
|
|
|
assert result2["type"] is FlowResultType.CREATE_ENTRY
|
|
assert result2["data"] == {
|
|
"api_key": "bla",
|
|
}
|
|
assert result2["options"] == RECOMMENDED_OPTIONS
|
|
assert len(mock_setup_entry.mock_calls) == 1
|
|
|
|
|
|
async def test_options(
|
|
hass: HomeAssistant, mock_config_entry, mock_init_component
|
|
) -> None:
|
|
"""Test the options form."""
|
|
options_flow = await hass.config_entries.options.async_init(
|
|
mock_config_entry.entry_id
|
|
)
|
|
options = await hass.config_entries.options.async_configure(
|
|
options_flow["flow_id"],
|
|
{
|
|
"prompt": "Speak like a pirate",
|
|
"max_tokens": 200,
|
|
},
|
|
)
|
|
await hass.async_block_till_done()
|
|
assert options["type"] is FlowResultType.CREATE_ENTRY
|
|
assert options["data"]["prompt"] == "Speak like a pirate"
|
|
assert options["data"]["max_tokens"] == 200
|
|
assert options["data"][CONF_CHAT_MODEL] == RECOMMENDED_CHAT_MODEL
|
|
|
|
|
|
@pytest.mark.parametrize(
|
|
("side_effect", "error"),
|
|
[
|
|
(APIConnectionError(request=None), "cannot_connect"),
|
|
(APITimeoutError(request=None), "timeout_connect"),
|
|
(
|
|
BadRequestError(
|
|
message="Your credit balance is too low to access the Claude API. Please go to Plans & Billing to upgrade or purchase credits.",
|
|
response=Response(
|
|
status_code=400,
|
|
request=Request(method="POST", url=URL()),
|
|
),
|
|
body={"type": "error", "error": {"type": "invalid_request_error"}},
|
|
),
|
|
"unknown",
|
|
),
|
|
(
|
|
AuthenticationError(
|
|
message="invalid x-api-key",
|
|
response=Response(
|
|
status_code=401,
|
|
request=Request(method="POST", url=URL()),
|
|
),
|
|
body={"type": "error", "error": {"type": "authentication_error"}},
|
|
),
|
|
"authentication_error",
|
|
),
|
|
(
|
|
InternalServerError(
|
|
message=None,
|
|
response=Response(
|
|
status_code=500,
|
|
request=Request(method="POST", url=URL()),
|
|
),
|
|
body=None,
|
|
),
|
|
"unknown",
|
|
),
|
|
(
|
|
APIResponseValidationError(
|
|
response=Response(
|
|
status_code=200,
|
|
request=Request(method="POST", url=URL()),
|
|
),
|
|
body=None,
|
|
),
|
|
"unknown",
|
|
),
|
|
],
|
|
)
|
|
async def test_form_invalid_auth(hass: HomeAssistant, side_effect, error) -> None:
|
|
"""Test we handle invalid auth."""
|
|
result = await hass.config_entries.flow.async_init(
|
|
DOMAIN, context={"source": config_entries.SOURCE_USER}
|
|
)
|
|
|
|
with patch(
|
|
"homeassistant.components.anthropic.config_flow.anthropic.resources.messages.AsyncMessages.create",
|
|
new_callable=AsyncMock,
|
|
side_effect=side_effect,
|
|
):
|
|
result2 = await hass.config_entries.flow.async_configure(
|
|
result["flow_id"],
|
|
{
|
|
"api_key": "bla",
|
|
},
|
|
)
|
|
|
|
assert result2["type"] is FlowResultType.FORM
|
|
assert result2["errors"] == {"base": error}
|
|
|
|
|
|
@pytest.mark.parametrize(
|
|
("current_options", "new_options", "expected_options"),
|
|
[
|
|
(
|
|
{
|
|
CONF_RECOMMENDED: True,
|
|
CONF_LLM_HASS_API: "none",
|
|
CONF_PROMPT: "bla",
|
|
},
|
|
{
|
|
CONF_RECOMMENDED: False,
|
|
CONF_PROMPT: "Speak like a pirate",
|
|
CONF_TEMPERATURE: 0.3,
|
|
},
|
|
{
|
|
CONF_RECOMMENDED: False,
|
|
CONF_PROMPT: "Speak like a pirate",
|
|
CONF_TEMPERATURE: 0.3,
|
|
CONF_CHAT_MODEL: RECOMMENDED_CHAT_MODEL,
|
|
CONF_MAX_TOKENS: RECOMMENDED_MAX_TOKENS,
|
|
},
|
|
),
|
|
(
|
|
{
|
|
CONF_RECOMMENDED: False,
|
|
CONF_PROMPT: "Speak like a pirate",
|
|
CONF_TEMPERATURE: 0.3,
|
|
CONF_CHAT_MODEL: RECOMMENDED_CHAT_MODEL,
|
|
CONF_MAX_TOKENS: RECOMMENDED_MAX_TOKENS,
|
|
},
|
|
{
|
|
CONF_RECOMMENDED: True,
|
|
CONF_LLM_HASS_API: "assist",
|
|
CONF_PROMPT: "",
|
|
},
|
|
{
|
|
CONF_RECOMMENDED: True,
|
|
CONF_LLM_HASS_API: "assist",
|
|
CONF_PROMPT: "",
|
|
},
|
|
),
|
|
],
|
|
)
|
|
async def test_options_switching(
|
|
hass: HomeAssistant,
|
|
mock_config_entry,
|
|
mock_init_component,
|
|
current_options,
|
|
new_options,
|
|
expected_options,
|
|
) -> None:
|
|
"""Test the options form."""
|
|
hass.config_entries.async_update_entry(mock_config_entry, options=current_options)
|
|
options_flow = await hass.config_entries.options.async_init(
|
|
mock_config_entry.entry_id
|
|
)
|
|
if current_options.get(CONF_RECOMMENDED) != new_options.get(CONF_RECOMMENDED):
|
|
options_flow = await hass.config_entries.options.async_configure(
|
|
options_flow["flow_id"],
|
|
{
|
|
**current_options,
|
|
CONF_RECOMMENDED: new_options[CONF_RECOMMENDED],
|
|
},
|
|
)
|
|
options = await hass.config_entries.options.async_configure(
|
|
options_flow["flow_id"],
|
|
new_options,
|
|
)
|
|
await hass.async_block_till_done()
|
|
assert options["type"] is FlowResultType.CREATE_ENTRY
|
|
assert options["data"] == expected_options
|