2019-03-24 15:16:50 +00:00
|
|
|
"""Test Axis config flow."""
|
|
|
|
from unittest.mock import Mock, patch
|
|
|
|
|
|
|
|
from homeassistant.components import axis
|
|
|
|
from homeassistant.components.axis import config_flow
|
|
|
|
|
2019-12-08 14:44:04 +00:00
|
|
|
from tests.common import MockConfigEntry, mock_coro
|
2019-03-24 15:16:50 +00:00
|
|
|
|
|
|
|
|
|
|
|
async def test_configured_devices(hass):
|
|
|
|
"""Test that configured devices works as expected."""
|
|
|
|
result = config_flow.configured_devices(hass)
|
|
|
|
|
|
|
|
assert not result
|
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
entry = MockConfigEntry(
|
|
|
|
domain=axis.DOMAIN, data={axis.config_flow.CONF_MAC: "1234"}
|
|
|
|
)
|
2019-03-24 15:16:50 +00:00
|
|
|
entry.add_to_hass(hass)
|
|
|
|
|
|
|
|
result = config_flow.configured_devices(hass)
|
|
|
|
|
|
|
|
assert len(result) == 1
|
|
|
|
|
|
|
|
|
|
|
|
async def test_flow_works(hass):
|
|
|
|
"""Test that config flow works."""
|
2019-07-31 19:25:30 +00:00
|
|
|
with patch("axis.AxisDevice") as mock_device:
|
|
|
|
|
|
|
|
def mock_constructor(loop, host, username, password, port, web_proto):
|
2019-03-24 15:16:50 +00:00
|
|
|
"""Fake the controller constructor."""
|
|
|
|
mock_device.loop = loop
|
|
|
|
mock_device.host = host
|
|
|
|
mock_device.username = username
|
|
|
|
mock_device.password = password
|
|
|
|
mock_device.port = port
|
|
|
|
return mock_device
|
|
|
|
|
|
|
|
mock_device.side_effect = mock_constructor
|
2019-07-31 19:25:30 +00:00
|
|
|
mock_device.vapix.params.system_serialnumber = "serialnumber"
|
|
|
|
mock_device.vapix.params.prodnbr = "prodnbr"
|
2019-08-12 14:42:12 +00:00
|
|
|
mock_device.vapix.params.prodtype = "prodtype"
|
|
|
|
mock_device.vapix.params.firmware_version = "firmware_version"
|
2019-03-24 15:16:50 +00:00
|
|
|
|
2019-04-15 22:06:45 +00:00
|
|
|
result = await hass.config_entries.flow.async_init(
|
2019-07-31 19:25:30 +00:00
|
|
|
config_flow.DOMAIN, context={"source": "user"}
|
2019-04-15 22:06:45 +00:00
|
|
|
)
|
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
assert result["type"] == "form"
|
|
|
|
assert result["step_id"] == "user"
|
2019-04-15 22:06:45 +00:00
|
|
|
|
|
|
|
result = await hass.config_entries.flow.async_configure(
|
2019-07-31 19:25:30 +00:00
|
|
|
result["flow_id"],
|
2019-04-15 22:06:45 +00:00
|
|
|
user_input={
|
2019-07-31 19:25:30 +00:00
|
|
|
config_flow.CONF_HOST: "1.2.3.4",
|
|
|
|
config_flow.CONF_USERNAME: "user",
|
|
|
|
config_flow.CONF_PASSWORD: "pass",
|
|
|
|
config_flow.CONF_PORT: 80,
|
|
|
|
},
|
2019-04-15 22:06:45 +00:00
|
|
|
)
|
2019-03-24 15:16:50 +00:00
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
assert result["type"] == "create_entry"
|
2020-01-02 23:02:59 +00:00
|
|
|
assert result["title"] == f"prodnbr - serialnumber"
|
2019-07-31 19:25:30 +00:00
|
|
|
assert result["data"] == {
|
2019-03-24 15:16:50 +00:00
|
|
|
axis.CONF_DEVICE: {
|
2019-07-31 19:25:30 +00:00
|
|
|
config_flow.CONF_HOST: "1.2.3.4",
|
|
|
|
config_flow.CONF_USERNAME: "user",
|
|
|
|
config_flow.CONF_PASSWORD: "pass",
|
|
|
|
config_flow.CONF_PORT: 80,
|
2019-03-24 15:16:50 +00:00
|
|
|
},
|
2019-07-31 19:25:30 +00:00
|
|
|
config_flow.CONF_MAC: "serialnumber",
|
|
|
|
config_flow.CONF_MODEL: "prodnbr",
|
|
|
|
config_flow.CONF_NAME: "prodnbr 0",
|
2019-03-24 15:16:50 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
async def test_flow_fails_already_configured(hass):
|
|
|
|
"""Test that config flow fails on already configured device."""
|
|
|
|
flow = config_flow.AxisFlowHandler()
|
|
|
|
flow.hass = hass
|
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
entry = MockConfigEntry(
|
|
|
|
domain=axis.DOMAIN, data={axis.config_flow.CONF_MAC: "1234"}
|
|
|
|
)
|
2019-03-24 15:16:50 +00:00
|
|
|
entry.add_to_hass(hass)
|
|
|
|
|
2019-04-02 18:13:11 +00:00
|
|
|
mock_device = Mock()
|
2019-07-31 19:25:30 +00:00
|
|
|
mock_device.vapix.params.system_serialnumber = "1234"
|
2019-04-02 18:13:11 +00:00
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
with patch(
|
|
|
|
"homeassistant.components.axis.config_flow.get_device",
|
|
|
|
return_value=mock_coro(mock_device),
|
|
|
|
):
|
|
|
|
result = await flow.async_step_user(
|
|
|
|
user_input={
|
|
|
|
config_flow.CONF_HOST: "1.2.3.4",
|
|
|
|
config_flow.CONF_USERNAME: "user",
|
|
|
|
config_flow.CONF_PASSWORD: "pass",
|
|
|
|
config_flow.CONF_PORT: 80,
|
|
|
|
}
|
|
|
|
)
|
2019-03-24 15:16:50 +00:00
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
assert result["errors"] == {"base": "already_configured"}
|
2019-03-24 15:16:50 +00:00
|
|
|
|
|
|
|
|
|
|
|
async def test_flow_fails_faulty_credentials(hass):
|
|
|
|
"""Test that config flow fails on faulty credentials."""
|
|
|
|
flow = config_flow.AxisFlowHandler()
|
|
|
|
flow.hass = hass
|
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
with patch(
|
|
|
|
"homeassistant.components.axis.config_flow.get_device",
|
|
|
|
side_effect=config_flow.AuthenticationRequired,
|
|
|
|
):
|
|
|
|
result = await flow.async_step_user(
|
|
|
|
user_input={
|
|
|
|
config_flow.CONF_HOST: "1.2.3.4",
|
|
|
|
config_flow.CONF_USERNAME: "user",
|
|
|
|
config_flow.CONF_PASSWORD: "pass",
|
|
|
|
config_flow.CONF_PORT: 80,
|
|
|
|
}
|
|
|
|
)
|
2019-03-24 15:16:50 +00:00
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
assert result["errors"] == {"base": "faulty_credentials"}
|
2019-03-24 15:16:50 +00:00
|
|
|
|
|
|
|
|
|
|
|
async def test_flow_fails_device_unavailable(hass):
|
|
|
|
"""Test that config flow fails on device unavailable."""
|
|
|
|
flow = config_flow.AxisFlowHandler()
|
|
|
|
flow.hass = hass
|
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
with patch(
|
|
|
|
"homeassistant.components.axis.config_flow.get_device",
|
|
|
|
side_effect=config_flow.CannotConnect,
|
|
|
|
):
|
|
|
|
result = await flow.async_step_user(
|
|
|
|
user_input={
|
|
|
|
config_flow.CONF_HOST: "1.2.3.4",
|
|
|
|
config_flow.CONF_USERNAME: "user",
|
|
|
|
config_flow.CONF_PASSWORD: "pass",
|
|
|
|
config_flow.CONF_PORT: 80,
|
|
|
|
}
|
|
|
|
)
|
2019-03-24 15:16:50 +00:00
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
assert result["errors"] == {"base": "device_unavailable"}
|
2019-03-24 15:16:50 +00:00
|
|
|
|
|
|
|
|
|
|
|
async def test_flow_create_entry(hass):
|
|
|
|
"""Test that create entry can generate a name without other entries."""
|
|
|
|
flow = config_flow.AxisFlowHandler()
|
|
|
|
flow.hass = hass
|
2019-07-31 19:25:30 +00:00
|
|
|
flow.model = "model"
|
2019-03-24 15:16:50 +00:00
|
|
|
|
|
|
|
result = await flow._create_entry()
|
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
assert result["data"][config_flow.CONF_NAME] == "model 0"
|
2019-03-24 15:16:50 +00:00
|
|
|
|
|
|
|
|
|
|
|
async def test_flow_create_entry_more_entries(hass):
|
|
|
|
"""Test that create entry can generate a name with other entries."""
|
|
|
|
entry = MockConfigEntry(
|
2019-07-31 19:25:30 +00:00
|
|
|
domain=axis.DOMAIN,
|
|
|
|
data={config_flow.CONF_NAME: "model 0", config_flow.CONF_MODEL: "model"},
|
|
|
|
)
|
2019-03-24 15:16:50 +00:00
|
|
|
entry.add_to_hass(hass)
|
|
|
|
entry2 = MockConfigEntry(
|
2019-07-31 19:25:30 +00:00
|
|
|
domain=axis.DOMAIN,
|
|
|
|
data={config_flow.CONF_NAME: "model 1", config_flow.CONF_MODEL: "model"},
|
|
|
|
)
|
2019-03-24 15:16:50 +00:00
|
|
|
entry2.add_to_hass(hass)
|
|
|
|
|
|
|
|
flow = config_flow.AxisFlowHandler()
|
|
|
|
flow.hass = hass
|
2019-07-31 19:25:30 +00:00
|
|
|
flow.model = "model"
|
2019-03-24 15:16:50 +00:00
|
|
|
|
|
|
|
result = await flow._create_entry()
|
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
assert result["data"][config_flow.CONF_NAME] == "model 2"
|
2019-03-24 15:16:50 +00:00
|
|
|
|
|
|
|
|
2019-05-21 22:36:26 +00:00
|
|
|
async def test_zeroconf_flow(hass):
|
|
|
|
"""Test that zeroconf discovery for new devices work."""
|
2019-07-31 19:25:30 +00:00
|
|
|
with patch.object(axis, "get_device", return_value=mock_coro(Mock())):
|
2019-04-15 22:06:45 +00:00
|
|
|
result = await hass.config_entries.flow.async_init(
|
|
|
|
config_flow.DOMAIN,
|
|
|
|
data={
|
2019-07-31 19:25:30 +00:00
|
|
|
config_flow.CONF_HOST: "1.2.3.4",
|
2019-04-15 22:06:45 +00:00
|
|
|
config_flow.CONF_PORT: 80,
|
2019-10-16 18:45:03 +00:00
|
|
|
"hostname": "name",
|
2019-07-31 19:25:30 +00:00
|
|
|
"properties": {"macaddress": "00408C12345"},
|
2019-04-15 22:06:45 +00:00
|
|
|
},
|
2019-07-31 19:25:30 +00:00
|
|
|
context={"source": "zeroconf"},
|
2019-04-15 22:06:45 +00:00
|
|
|
)
|
2019-03-24 15:16:50 +00:00
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
assert result["type"] == "form"
|
|
|
|
assert result["step_id"] == "user"
|
2019-03-24 15:16:50 +00:00
|
|
|
|
|
|
|
|
2019-05-21 22:36:26 +00:00
|
|
|
async def test_zeroconf_flow_already_configured(hass):
|
|
|
|
"""Test that zeroconf doesn't setup already configured devices."""
|
2019-04-02 18:13:11 +00:00
|
|
|
entry = MockConfigEntry(
|
|
|
|
domain=axis.DOMAIN,
|
2019-07-31 19:25:30 +00:00
|
|
|
data={
|
|
|
|
axis.CONF_DEVICE: {axis.config_flow.CONF_HOST: "1.2.3.4"},
|
|
|
|
axis.config_flow.CONF_MAC: "00408C12345",
|
|
|
|
},
|
2019-04-02 18:13:11 +00:00
|
|
|
)
|
2019-03-24 15:16:50 +00:00
|
|
|
entry.add_to_hass(hass)
|
|
|
|
|
2019-04-15 22:06:45 +00:00
|
|
|
result = await hass.config_entries.flow.async_init(
|
|
|
|
config_flow.DOMAIN,
|
|
|
|
data={
|
2019-07-31 19:25:30 +00:00
|
|
|
config_flow.CONF_HOST: "1.2.3.4",
|
|
|
|
config_flow.CONF_USERNAME: "user",
|
|
|
|
config_flow.CONF_PASSWORD: "pass",
|
2019-04-15 22:06:45 +00:00
|
|
|
config_flow.CONF_PORT: 80,
|
2019-07-31 19:25:30 +00:00
|
|
|
"hostname": "name",
|
|
|
|
"properties": {"macaddress": "00408C12345"},
|
2019-04-15 22:06:45 +00:00
|
|
|
},
|
2019-07-31 19:25:30 +00:00
|
|
|
context={"source": "zeroconf"},
|
2019-04-15 22:06:45 +00:00
|
|
|
)
|
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
assert result["type"] == "abort"
|
|
|
|
assert result["reason"] == "already_configured"
|
2019-03-24 15:16:50 +00:00
|
|
|
|
|
|
|
|
2019-06-10 16:12:17 +00:00
|
|
|
async def test_zeroconf_flow_ignore_non_axis_device(hass):
|
|
|
|
"""Test that zeroconf doesn't setup devices with link local addresses."""
|
|
|
|
result = await hass.config_entries.flow.async_init(
|
|
|
|
config_flow.DOMAIN,
|
|
|
|
data={
|
2019-07-31 19:25:30 +00:00
|
|
|
config_flow.CONF_HOST: "169.254.3.4",
|
|
|
|
"properties": {"macaddress": "01234567890"},
|
2019-06-10 16:12:17 +00:00
|
|
|
},
|
2019-07-31 19:25:30 +00:00
|
|
|
context={"source": "zeroconf"},
|
2019-06-10 16:12:17 +00:00
|
|
|
)
|
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
assert result["type"] == "abort"
|
|
|
|
assert result["reason"] == "not_axis_device"
|
2019-06-10 16:12:17 +00:00
|
|
|
|
|
|
|
|
2019-05-21 22:36:26 +00:00
|
|
|
async def test_zeroconf_flow_ignore_link_local_address(hass):
|
|
|
|
"""Test that zeroconf doesn't setup devices with link local addresses."""
|
2019-04-15 22:06:45 +00:00
|
|
|
result = await hass.config_entries.flow.async_init(
|
|
|
|
config_flow.DOMAIN,
|
2019-06-10 16:12:17 +00:00
|
|
|
data={
|
2019-07-31 19:25:30 +00:00
|
|
|
config_flow.CONF_HOST: "169.254.3.4",
|
|
|
|
"properties": {"macaddress": "00408C12345"},
|
2019-06-10 16:12:17 +00:00
|
|
|
},
|
2019-07-31 19:25:30 +00:00
|
|
|
context={"source": "zeroconf"},
|
2019-04-15 22:06:45 +00:00
|
|
|
)
|
2019-03-24 15:16:50 +00:00
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
assert result["type"] == "abort"
|
|
|
|
assert result["reason"] == "link_local_address"
|