2020-08-12 13:09:47 +00:00
|
|
|
"""Test the roon config flow."""
|
2021-01-01 21:31:56 +00:00
|
|
|
from unittest.mock import patch
|
|
|
|
|
2022-04-18 14:27:14 +00:00
|
|
|
from homeassistant import config_entries, data_entry_flow
|
2020-08-12 13:09:47 +00:00
|
|
|
from homeassistant.components.roon.const import DOMAIN
|
2023-02-08 16:12:54 +00:00
|
|
|
from homeassistant.core import HomeAssistant
|
2020-08-12 13:09:47 +00:00
|
|
|
|
2022-04-18 14:27:14 +00:00
|
|
|
from tests.common import MockConfigEntry
|
|
|
|
|
2020-08-12 13:09:47 +00:00
|
|
|
|
|
|
|
class RoonApiMock:
|
2021-01-18 15:00:30 +00:00
|
|
|
"""Class to mock the Roon API for testing."""
|
|
|
|
|
|
|
|
@property
|
|
|
|
def token(self):
|
|
|
|
"""Return a good authentication key."""
|
|
|
|
return "good_token"
|
|
|
|
|
|
|
|
@property
|
|
|
|
def core_id(self):
|
|
|
|
"""Return the roon host."""
|
|
|
|
return "core_id"
|
|
|
|
|
2022-04-18 14:27:14 +00:00
|
|
|
@property
|
|
|
|
def core_name(self):
|
|
|
|
"""Return the roon core name."""
|
|
|
|
return "Roon Core"
|
|
|
|
|
2021-01-18 15:00:30 +00:00
|
|
|
def stop(self):
|
2022-04-18 14:27:14 +00:00
|
|
|
"""Stop socket."""
|
2021-01-18 15:00:30 +00:00
|
|
|
return
|
|
|
|
|
|
|
|
|
|
|
|
class RoonApiMockNoToken(RoonApiMock):
|
|
|
|
"""Class to mock the Roon API for testing, with failed authorisation."""
|
|
|
|
|
|
|
|
@property
|
|
|
|
def token(self):
|
|
|
|
"""Return a bad authentication key."""
|
|
|
|
return None
|
|
|
|
|
2020-08-12 13:09:47 +00:00
|
|
|
|
2021-01-18 15:00:30 +00:00
|
|
|
class RoonApiMockException(RoonApiMock):
|
|
|
|
"""Class to mock the Roon API for testing, throws an unexpected exception."""
|
2020-08-12 13:09:47 +00:00
|
|
|
|
|
|
|
@property
|
|
|
|
def token(self):
|
2021-01-18 15:00:30 +00:00
|
|
|
"""Throw exception."""
|
|
|
|
raise Exception
|
|
|
|
|
2020-08-12 13:09:47 +00:00
|
|
|
|
2021-01-18 15:00:30 +00:00
|
|
|
class RoonDiscoveryMock:
|
|
|
|
"""Class to mock Roon Discovery for testing."""
|
|
|
|
|
|
|
|
def all(self):
|
|
|
|
"""Return a discovered roon server."""
|
|
|
|
return ["2.2.2.2"]
|
|
|
|
|
|
|
|
def stop(self):
|
|
|
|
"""Stop discovery running."""
|
2020-08-12 13:09:47 +00:00
|
|
|
return
|
|
|
|
|
|
|
|
|
2021-01-18 15:00:30 +00:00
|
|
|
class RoonDiscoveryFailedMock(RoonDiscoveryMock):
|
|
|
|
"""Class to mock Roon Discovery for testing, with no servers discovered."""
|
|
|
|
|
|
|
|
def all(self):
|
|
|
|
"""Return no discovered roon servers."""
|
|
|
|
return []
|
|
|
|
|
|
|
|
|
2023-02-08 16:12:54 +00:00
|
|
|
async def test_successful_discovery_and_auth(hass: HomeAssistant) -> None:
|
2021-01-18 15:00:30 +00:00
|
|
|
"""Test when discovery and auth both work ok."""
|
|
|
|
|
|
|
|
with patch(
|
2020-08-12 13:09:47 +00:00
|
|
|
"homeassistant.components.roon.config_flow.RoonApi",
|
2021-01-18 15:00:30 +00:00
|
|
|
return_value=RoonApiMock(),
|
|
|
|
), patch(
|
|
|
|
"homeassistant.components.roon.config_flow.RoonDiscovery",
|
|
|
|
return_value=RoonDiscoveryMock(),
|
|
|
|
), patch(
|
2020-08-27 11:56:20 +00:00
|
|
|
"homeassistant.components.roon.async_setup_entry",
|
|
|
|
return_value=True,
|
2021-01-18 15:00:30 +00:00
|
|
|
):
|
|
|
|
result = await hass.config_entries.flow.async_init(
|
|
|
|
DOMAIN, context={"source": config_entries.SOURCE_USER}
|
2020-08-12 13:09:47 +00:00
|
|
|
)
|
2021-01-18 15:00:30 +00:00
|
|
|
await hass.async_block_till_done()
|
|
|
|
|
|
|
|
# Should go straight to link if server was discovered
|
|
|
|
assert result["type"] == "form"
|
|
|
|
assert result["step_id"] == "link"
|
|
|
|
assert result["errors"] == {}
|
|
|
|
|
2020-08-12 13:09:47 +00:00
|
|
|
result2 = await hass.config_entries.flow.async_configure(
|
|
|
|
result["flow_id"], user_input={}
|
|
|
|
)
|
2020-10-24 14:20:56 +00:00
|
|
|
await hass.async_block_till_done()
|
2020-08-12 13:09:47 +00:00
|
|
|
|
|
|
|
assert result2["title"] == "Roon Labs Music Player"
|
2021-01-18 15:00:30 +00:00
|
|
|
assert result2["data"] == {
|
|
|
|
"host": None,
|
2022-04-18 14:27:14 +00:00
|
|
|
"port": None,
|
2021-01-18 15:00:30 +00:00
|
|
|
"api_key": "good_token",
|
|
|
|
"roon_server_id": "core_id",
|
2022-04-18 14:27:14 +00:00
|
|
|
"roon_server_name": "Roon Core",
|
2021-01-18 15:00:30 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2023-02-08 16:12:54 +00:00
|
|
|
async def test_unsuccessful_discovery_user_form_and_auth(hass: HomeAssistant) -> None:
|
2021-01-18 15:00:30 +00:00
|
|
|
"""Test unsuccessful discover, user adding the host via the form and then successful auth."""
|
|
|
|
|
|
|
|
with patch(
|
2020-08-12 13:09:47 +00:00
|
|
|
"homeassistant.components.roon.config_flow.RoonApi",
|
2021-01-18 15:00:30 +00:00
|
|
|
return_value=RoonApiMock(),
|
|
|
|
), patch(
|
|
|
|
"homeassistant.components.roon.config_flow.RoonDiscovery",
|
|
|
|
return_value=RoonDiscoveryFailedMock(),
|
|
|
|
), patch(
|
|
|
|
"homeassistant.components.roon.async_setup_entry",
|
|
|
|
return_value=True,
|
2020-08-12 13:09:47 +00:00
|
|
|
):
|
2021-01-18 15:00:30 +00:00
|
|
|
result = await hass.config_entries.flow.async_init(
|
|
|
|
DOMAIN, context={"source": config_entries.SOURCE_USER}
|
|
|
|
)
|
|
|
|
await hass.async_block_till_done()
|
|
|
|
|
|
|
|
# Should show the form if server was not discovered
|
|
|
|
assert result["type"] == "form"
|
2022-04-18 14:27:14 +00:00
|
|
|
assert result["step_id"] == "fallback"
|
2021-01-18 15:00:30 +00:00
|
|
|
assert result["errors"] == {}
|
|
|
|
|
2020-08-12 13:09:47 +00:00
|
|
|
await hass.config_entries.flow.async_configure(
|
2022-04-18 14:27:14 +00:00
|
|
|
result["flow_id"], {"host": "1.1.1.1", "port": 9331}
|
2020-08-12 13:09:47 +00:00
|
|
|
)
|
|
|
|
result2 = await hass.config_entries.flow.async_configure(
|
|
|
|
result["flow_id"], user_input={}
|
|
|
|
)
|
2021-01-18 15:00:30 +00:00
|
|
|
await hass.async_block_till_done()
|
2020-08-12 13:09:47 +00:00
|
|
|
|
2021-01-18 15:00:30 +00:00
|
|
|
assert result2["title"] == "Roon Labs Music Player"
|
|
|
|
assert result2["data"] == {
|
|
|
|
"host": "1.1.1.1",
|
|
|
|
"api_key": "good_token",
|
2022-04-18 14:27:14 +00:00
|
|
|
"port": 9331,
|
2021-01-18 15:00:30 +00:00
|
|
|
"roon_server_id": "core_id",
|
2022-04-18 14:27:14 +00:00
|
|
|
"roon_server_name": "Roon Core",
|
2021-01-18 15:00:30 +00:00
|
|
|
}
|
2020-08-12 13:09:47 +00:00
|
|
|
|
|
|
|
|
2023-02-08 16:12:54 +00:00
|
|
|
async def test_duplicate_config(hass: HomeAssistant) -> None:
|
2022-04-18 14:27:14 +00:00
|
|
|
"""Test user adding the host via the form for host that is already configured."""
|
|
|
|
|
|
|
|
CONFIG = {"host": "1.1.1.1"}
|
|
|
|
|
|
|
|
MockConfigEntry(domain=DOMAIN, unique_id="0123456789", data=CONFIG).add_to_hass(
|
|
|
|
hass
|
|
|
|
)
|
|
|
|
|
|
|
|
with patch(
|
|
|
|
"homeassistant.components.roon.config_flow.RoonApi",
|
|
|
|
return_value=RoonApiMock(),
|
|
|
|
), patch(
|
|
|
|
"homeassistant.components.roon.config_flow.RoonDiscovery",
|
|
|
|
return_value=RoonDiscoveryFailedMock(),
|
|
|
|
):
|
|
|
|
result = await hass.config_entries.flow.async_init(
|
|
|
|
DOMAIN, context={"source": config_entries.SOURCE_USER}
|
|
|
|
)
|
|
|
|
await hass.async_block_till_done()
|
|
|
|
|
|
|
|
# Should show the form if server was not discovered
|
|
|
|
assert result["type"] == "form"
|
|
|
|
assert result["step_id"] == "fallback"
|
|
|
|
assert result["errors"] == {}
|
|
|
|
|
|
|
|
await hass.config_entries.flow.async_configure(
|
|
|
|
result["flow_id"], {"host": "1.1.1.1", "port": 9331}
|
|
|
|
)
|
|
|
|
result2 = await hass.config_entries.flow.async_configure(
|
|
|
|
result["flow_id"], user_input={}
|
|
|
|
)
|
|
|
|
await hass.async_block_till_done()
|
|
|
|
|
2022-07-07 16:57:36 +00:00
|
|
|
assert result2["type"] == data_entry_flow.FlowResultType.ABORT
|
2022-04-18 14:27:14 +00:00
|
|
|
assert result2["reason"] == "already_configured"
|
|
|
|
|
|
|
|
|
2023-02-08 16:12:54 +00:00
|
|
|
async def test_successful_discovery_no_auth(hass: HomeAssistant) -> None:
|
2021-01-18 15:00:30 +00:00
|
|
|
"""Test successful discover, but failed auth."""
|
2020-08-12 13:09:47 +00:00
|
|
|
|
|
|
|
with patch(
|
2020-08-27 11:56:20 +00:00
|
|
|
"homeassistant.components.roon.config_flow.RoonApi",
|
2021-01-18 15:00:30 +00:00
|
|
|
return_value=RoonApiMockNoToken(),
|
|
|
|
), patch(
|
|
|
|
"homeassistant.components.roon.config_flow.RoonDiscovery",
|
|
|
|
return_value=RoonDiscoveryMock(),
|
|
|
|
), patch(
|
|
|
|
"homeassistant.components.roon.config_flow.TIMEOUT",
|
|
|
|
0,
|
|
|
|
), patch(
|
|
|
|
"homeassistant.components.roon.config_flow.AUTHENTICATE_TIMEOUT",
|
|
|
|
0.01,
|
|
|
|
), patch(
|
|
|
|
"homeassistant.components.roon.async_setup_entry",
|
|
|
|
return_value=True,
|
2020-08-12 13:09:47 +00:00
|
|
|
):
|
2021-01-18 15:00:30 +00:00
|
|
|
result = await hass.config_entries.flow.async_init(
|
|
|
|
DOMAIN, context={"source": config_entries.SOURCE_USER}
|
2020-08-12 13:09:47 +00:00
|
|
|
)
|
2021-01-18 15:00:30 +00:00
|
|
|
await hass.async_block_till_done()
|
|
|
|
|
|
|
|
# Should go straight to link if server was discovered
|
|
|
|
assert result["type"] == "form"
|
|
|
|
assert result["step_id"] == "link"
|
|
|
|
assert result["errors"] == {}
|
2020-08-12 13:09:47 +00:00
|
|
|
|
|
|
|
result2 = await hass.config_entries.flow.async_configure(
|
|
|
|
result["flow_id"], user_input={}
|
|
|
|
)
|
2021-01-18 15:00:30 +00:00
|
|
|
await hass.async_block_till_done()
|
2020-08-12 13:09:47 +00:00
|
|
|
|
2021-01-18 15:00:30 +00:00
|
|
|
assert result2["errors"] == {"base": "invalid_auth"}
|
2020-08-12 13:09:47 +00:00
|
|
|
|
|
|
|
|
2023-02-08 16:12:54 +00:00
|
|
|
async def test_unexpected_exception(hass: HomeAssistant) -> None:
|
2021-01-18 15:00:30 +00:00
|
|
|
"""Test successful discover, and unexpected exception during auth."""
|
2020-08-12 13:09:47 +00:00
|
|
|
|
2021-01-18 15:00:30 +00:00
|
|
|
with patch(
|
2020-08-12 13:09:47 +00:00
|
|
|
"homeassistant.components.roon.config_flow.RoonApi",
|
2021-01-18 15:00:30 +00:00
|
|
|
return_value=RoonApiMockException(),
|
|
|
|
), patch(
|
|
|
|
"homeassistant.components.roon.config_flow.RoonDiscovery",
|
|
|
|
return_value=RoonDiscoveryMock(),
|
|
|
|
), patch(
|
2020-08-27 11:56:20 +00:00
|
|
|
"homeassistant.components.roon.async_setup_entry",
|
|
|
|
return_value=True,
|
2021-01-18 15:00:30 +00:00
|
|
|
):
|
|
|
|
result = await hass.config_entries.flow.async_init(
|
|
|
|
DOMAIN, context={"source": config_entries.SOURCE_USER}
|
2020-08-12 13:09:47 +00:00
|
|
|
)
|
2021-01-18 15:00:30 +00:00
|
|
|
await hass.async_block_till_done()
|
|
|
|
|
|
|
|
# Should go straight to link if server was discovered
|
|
|
|
assert result["type"] == "form"
|
|
|
|
assert result["step_id"] == "link"
|
|
|
|
assert result["errors"] == {}
|
|
|
|
|
2020-08-12 13:09:47 +00:00
|
|
|
result2 = await hass.config_entries.flow.async_configure(
|
|
|
|
result["flow_id"], user_input={}
|
|
|
|
)
|
2020-10-24 14:20:56 +00:00
|
|
|
await hass.async_block_till_done()
|
2020-08-12 13:09:47 +00:00
|
|
|
|
2021-01-18 15:00:30 +00:00
|
|
|
assert result2["errors"] == {"base": "unknown"}
|