2019-03-24 16:16:50 +01:00
|
|
|
"""Test Axis config flow."""
|
|
|
|
from homeassistant.components import axis
|
|
|
|
from homeassistant.components.axis import config_flow
|
2020-05-14 10:49:27 +02:00
|
|
|
from homeassistant.components.axis.const import CONF_MODEL, DOMAIN as AXIS_DOMAIN
|
|
|
|
from homeassistant.const import (
|
|
|
|
CONF_HOST,
|
|
|
|
CONF_MAC,
|
|
|
|
CONF_NAME,
|
|
|
|
CONF_PASSWORD,
|
|
|
|
CONF_PORT,
|
|
|
|
CONF_USERNAME,
|
|
|
|
)
|
2019-03-24 16:16:50 +01:00
|
|
|
|
2020-01-30 22:20:30 +01:00
|
|
|
from .test_device import MAC, MODEL, NAME, setup_axis_integration
|
2019-03-24 16:16:50 +01:00
|
|
|
|
2020-04-30 13:29:50 -07:00
|
|
|
from tests.async_mock import Mock, patch
|
2020-04-25 14:32:55 -07:00
|
|
|
from tests.common import MockConfigEntry
|
2019-03-24 16:16:50 +01:00
|
|
|
|
|
|
|
|
2020-01-05 10:11:17 +01:00
|
|
|
def setup_mock_axis_device(mock_device):
|
|
|
|
"""Prepare mock axis device."""
|
|
|
|
|
2020-05-14 10:49:27 +02:00
|
|
|
def mock_constructor(host, username, password, port, web_proto):
|
2020-01-05 10:11:17 +01:00
|
|
|
"""Fake the controller constructor."""
|
|
|
|
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
|
|
|
|
mock_device.vapix.params.system_serialnumber = MAC
|
|
|
|
mock_device.vapix.params.prodnbr = "prodnbr"
|
|
|
|
mock_device.vapix.params.prodtype = "prodtype"
|
|
|
|
mock_device.vapix.params.firmware_version = "firmware_version"
|
|
|
|
|
|
|
|
|
|
|
|
async def test_flow_manual_configuration(hass):
|
2019-03-24 16:16:50 +01:00
|
|
|
"""Test that config flow works."""
|
2020-01-05 10:11:17 +01:00
|
|
|
result = await hass.config_entries.flow.async_init(
|
2020-05-14 10:49:27 +02:00
|
|
|
AXIS_DOMAIN, context={"source": "user"}
|
2020-01-05 10:11:17 +01:00
|
|
|
)
|
2019-07-31 12:25:30 -07:00
|
|
|
|
2020-01-05 10:11:17 +01:00
|
|
|
assert result["type"] == "form"
|
|
|
|
assert result["step_id"] == "user"
|
2019-03-24 16:16:50 +01:00
|
|
|
|
2020-01-05 10:11:17 +01:00
|
|
|
with patch("axis.AxisDevice") as mock_device:
|
2019-04-16 00:06:45 +02:00
|
|
|
|
2020-01-05 10:11:17 +01:00
|
|
|
setup_mock_axis_device(mock_device)
|
2019-04-16 00:06:45 +02:00
|
|
|
|
|
|
|
result = await hass.config_entries.flow.async_configure(
|
2019-07-31 12:25:30 -07:00
|
|
|
result["flow_id"],
|
2019-04-16 00:06:45 +02:00
|
|
|
user_input={
|
2020-05-14 10:49:27 +02:00
|
|
|
CONF_HOST: "1.2.3.4",
|
|
|
|
CONF_USERNAME: "user",
|
|
|
|
CONF_PASSWORD: "pass",
|
|
|
|
CONF_PORT: 80,
|
2019-07-31 12:25:30 -07:00
|
|
|
},
|
2019-04-16 00:06:45 +02:00
|
|
|
)
|
2019-03-24 16:16:50 +01:00
|
|
|
|
2019-07-31 12:25:30 -07:00
|
|
|
assert result["type"] == "create_entry"
|
2020-01-05 10:11:17 +01:00
|
|
|
assert result["title"] == f"prodnbr - {MAC}"
|
2019-07-31 12:25:30 -07:00
|
|
|
assert result["data"] == {
|
2020-05-14 10:49:27 +02:00
|
|
|
CONF_HOST: "1.2.3.4",
|
|
|
|
CONF_USERNAME: "user",
|
|
|
|
CONF_PASSWORD: "pass",
|
|
|
|
CONF_PORT: 80,
|
|
|
|
CONF_MAC: MAC,
|
|
|
|
CONF_MODEL: "prodnbr",
|
|
|
|
CONF_NAME: "prodnbr 0",
|
2019-03-24 16:16:50 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2020-01-05 10:11:17 +01:00
|
|
|
async def test_manual_configuration_update_configuration(hass):
|
|
|
|
"""Test that config flow fails on already configured device."""
|
|
|
|
device = await setup_axis_integration(hass)
|
|
|
|
|
|
|
|
result = await hass.config_entries.flow.async_init(
|
2020-05-14 10:49:27 +02:00
|
|
|
AXIS_DOMAIN, context={"source": "user"}
|
2020-01-05 10:11:17 +01:00
|
|
|
)
|
|
|
|
|
|
|
|
assert result["type"] == "form"
|
|
|
|
assert result["step_id"] == "user"
|
|
|
|
|
|
|
|
mock_device = Mock()
|
|
|
|
mock_device.vapix.params.system_serialnumber = MAC
|
|
|
|
|
|
|
|
with patch(
|
|
|
|
"homeassistant.components.axis.config_flow.get_device",
|
2020-04-25 14:32:55 -07:00
|
|
|
return_value=mock_device,
|
2020-01-05 10:11:17 +01:00
|
|
|
):
|
|
|
|
result = await hass.config_entries.flow.async_configure(
|
|
|
|
result["flow_id"],
|
|
|
|
user_input={
|
2020-05-14 10:49:27 +02:00
|
|
|
CONF_HOST: "2.3.4.5",
|
|
|
|
CONF_USERNAME: "user",
|
|
|
|
CONF_PASSWORD: "pass",
|
|
|
|
CONF_PORT: 80,
|
2020-01-05 10:11:17 +01:00
|
|
|
},
|
|
|
|
)
|
|
|
|
|
|
|
|
assert result["type"] == "abort"
|
2020-01-30 22:20:30 +01:00
|
|
|
assert result["reason"] == "already_configured"
|
2020-05-14 10:49:27 +02:00
|
|
|
assert device.host == "2.3.4.5"
|
2020-01-05 10:11:17 +01:00
|
|
|
|
|
|
|
|
2019-03-24 16:16:50 +01:00
|
|
|
async def test_flow_fails_already_configured(hass):
|
|
|
|
"""Test that config flow fails on already configured device."""
|
2020-01-04 08:58:18 +01:00
|
|
|
await setup_axis_integration(hass)
|
|
|
|
|
2020-01-05 10:11:17 +01:00
|
|
|
result = await hass.config_entries.flow.async_init(
|
2020-05-14 10:49:27 +02:00
|
|
|
AXIS_DOMAIN, context={"source": "user"}
|
2020-01-05 10:11:17 +01:00
|
|
|
)
|
|
|
|
|
|
|
|
assert result["type"] == "form"
|
|
|
|
assert result["step_id"] == "user"
|
2019-03-24 16:16:50 +01:00
|
|
|
|
2019-04-02 20:13:11 +02:00
|
|
|
mock_device = Mock()
|
2020-01-04 08:58:18 +01:00
|
|
|
mock_device.vapix.params.system_serialnumber = MAC
|
2019-04-02 20:13:11 +02:00
|
|
|
|
2019-07-31 12:25:30 -07:00
|
|
|
with patch(
|
|
|
|
"homeassistant.components.axis.config_flow.get_device",
|
2020-04-25 14:32:55 -07:00
|
|
|
return_value=mock_device,
|
2020-01-05 10:11:17 +01:00
|
|
|
):
|
|
|
|
result = await hass.config_entries.flow.async_configure(
|
|
|
|
result["flow_id"],
|
2019-07-31 12:25:30 -07:00
|
|
|
user_input={
|
2020-05-14 10:49:27 +02:00
|
|
|
CONF_HOST: "1.2.3.4",
|
|
|
|
CONF_USERNAME: "user",
|
|
|
|
CONF_PASSWORD: "pass",
|
|
|
|
CONF_PORT: 80,
|
2020-01-05 10:11:17 +01:00
|
|
|
},
|
2019-07-31 12:25:30 -07:00
|
|
|
)
|
2019-03-24 16:16:50 +01:00
|
|
|
|
2020-01-05 10:11:17 +01:00
|
|
|
assert result["type"] == "abort"
|
|
|
|
assert result["reason"] == "already_configured"
|
|
|
|
|
2019-03-24 16:16:50 +01:00
|
|
|
|
|
|
|
async def test_flow_fails_faulty_credentials(hass):
|
|
|
|
"""Test that config flow fails on faulty credentials."""
|
2020-01-05 10:11:17 +01:00
|
|
|
result = await hass.config_entries.flow.async_init(
|
2020-05-14 10:49:27 +02:00
|
|
|
AXIS_DOMAIN, context={"source": "user"}
|
2020-01-05 10:11:17 +01:00
|
|
|
)
|
|
|
|
|
|
|
|
assert result["type"] == "form"
|
|
|
|
assert result["step_id"] == "user"
|
2019-03-24 16:16:50 +01:00
|
|
|
|
2019-07-31 12:25:30 -07:00
|
|
|
with patch(
|
|
|
|
"homeassistant.components.axis.config_flow.get_device",
|
|
|
|
side_effect=config_flow.AuthenticationRequired,
|
|
|
|
):
|
2020-01-05 10:11:17 +01:00
|
|
|
result = await hass.config_entries.flow.async_configure(
|
|
|
|
result["flow_id"],
|
2019-07-31 12:25:30 -07:00
|
|
|
user_input={
|
2020-05-14 10:49:27 +02:00
|
|
|
CONF_HOST: "1.2.3.4",
|
|
|
|
CONF_USERNAME: "user",
|
|
|
|
CONF_PASSWORD: "pass",
|
|
|
|
CONF_PORT: 80,
|
2020-01-05 10:11:17 +01:00
|
|
|
},
|
2019-07-31 12:25:30 -07:00
|
|
|
)
|
2019-03-24 16:16:50 +01:00
|
|
|
|
2019-07-31 12:25:30 -07:00
|
|
|
assert result["errors"] == {"base": "faulty_credentials"}
|
2019-03-24 16:16:50 +01:00
|
|
|
|
|
|
|
|
|
|
|
async def test_flow_fails_device_unavailable(hass):
|
|
|
|
"""Test that config flow fails on device unavailable."""
|
2020-01-05 10:11:17 +01:00
|
|
|
result = await hass.config_entries.flow.async_init(
|
2020-05-14 10:49:27 +02:00
|
|
|
AXIS_DOMAIN, context={"source": "user"}
|
2020-01-05 10:11:17 +01:00
|
|
|
)
|
|
|
|
|
|
|
|
assert result["type"] == "form"
|
|
|
|
assert result["step_id"] == "user"
|
2019-03-24 16:16:50 +01:00
|
|
|
|
2019-07-31 12:25:30 -07:00
|
|
|
with patch(
|
|
|
|
"homeassistant.components.axis.config_flow.get_device",
|
|
|
|
side_effect=config_flow.CannotConnect,
|
|
|
|
):
|
2020-01-05 10:11:17 +01:00
|
|
|
result = await hass.config_entries.flow.async_configure(
|
|
|
|
result["flow_id"],
|
2019-07-31 12:25:30 -07:00
|
|
|
user_input={
|
2020-05-14 10:49:27 +02:00
|
|
|
CONF_HOST: "1.2.3.4",
|
|
|
|
CONF_USERNAME: "user",
|
|
|
|
CONF_PASSWORD: "pass",
|
|
|
|
CONF_PORT: 80,
|
2020-01-05 10:11:17 +01:00
|
|
|
},
|
2019-07-31 12:25:30 -07:00
|
|
|
)
|
2019-03-24 16:16:50 +01:00
|
|
|
|
2019-07-31 12:25:30 -07:00
|
|
|
assert result["errors"] == {"base": "device_unavailable"}
|
2019-03-24 16:16:50 +01:00
|
|
|
|
|
|
|
|
2020-01-05 10:11:17 +01:00
|
|
|
async def test_flow_create_entry_multiple_existing_entries_of_same_model(hass):
|
2019-03-24 16:16:50 +01:00
|
|
|
"""Test that create entry can generate a name with other entries."""
|
|
|
|
entry = MockConfigEntry(
|
2020-05-14 10:49:27 +02:00
|
|
|
domain=AXIS_DOMAIN, data={CONF_NAME: "prodnbr 0", CONF_MODEL: "prodnbr"},
|
2019-07-31 12:25:30 -07:00
|
|
|
)
|
2019-03-24 16:16:50 +01:00
|
|
|
entry.add_to_hass(hass)
|
|
|
|
entry2 = MockConfigEntry(
|
2020-05-14 10:49:27 +02:00
|
|
|
domain=AXIS_DOMAIN, data={CONF_NAME: "prodnbr 1", CONF_MODEL: "prodnbr"},
|
2019-07-31 12:25:30 -07:00
|
|
|
)
|
2019-03-24 16:16:50 +01:00
|
|
|
entry2.add_to_hass(hass)
|
|
|
|
|
2020-01-05 10:11:17 +01:00
|
|
|
result = await hass.config_entries.flow.async_init(
|
2020-05-14 10:49:27 +02:00
|
|
|
AXIS_DOMAIN, context={"source": "user"}
|
2020-01-05 10:11:17 +01:00
|
|
|
)
|
|
|
|
|
|
|
|
assert result["type"] == "form"
|
|
|
|
assert result["step_id"] == "user"
|
2019-03-24 16:16:50 +01:00
|
|
|
|
2020-01-05 10:11:17 +01:00
|
|
|
with patch("axis.AxisDevice") as mock_device:
|
2019-03-24 16:16:50 +01:00
|
|
|
|
2020-01-05 10:11:17 +01:00
|
|
|
setup_mock_axis_device(mock_device)
|
|
|
|
|
|
|
|
result = await hass.config_entries.flow.async_configure(
|
|
|
|
result["flow_id"],
|
|
|
|
user_input={
|
2020-05-14 10:49:27 +02:00
|
|
|
CONF_HOST: "1.2.3.4",
|
|
|
|
CONF_USERNAME: "user",
|
|
|
|
CONF_PASSWORD: "pass",
|
|
|
|
CONF_PORT: 80,
|
2020-01-05 10:11:17 +01:00
|
|
|
},
|
|
|
|
)
|
|
|
|
|
|
|
|
assert result["type"] == "create_entry"
|
|
|
|
assert result["title"] == f"prodnbr - {MAC}"
|
|
|
|
assert result["data"] == {
|
2020-05-14 10:49:27 +02:00
|
|
|
CONF_HOST: "1.2.3.4",
|
|
|
|
CONF_USERNAME: "user",
|
|
|
|
CONF_PASSWORD: "pass",
|
|
|
|
CONF_PORT: 80,
|
|
|
|
CONF_MAC: MAC,
|
|
|
|
CONF_MODEL: "prodnbr",
|
|
|
|
CONF_NAME: "prodnbr 2",
|
2020-01-05 10:11:17 +01:00
|
|
|
}
|
|
|
|
|
2020-05-14 10:49:27 +02:00
|
|
|
assert result["data"][CONF_NAME] == "prodnbr 2"
|
2019-03-24 16:16:50 +01:00
|
|
|
|
|
|
|
|
2019-05-22 00:36:26 +02:00
|
|
|
async def test_zeroconf_flow(hass):
|
|
|
|
"""Test that zeroconf discovery for new devices work."""
|
2020-05-14 10:49:27 +02:00
|
|
|
with patch.object(axis.device, "get_device", return_value=Mock()):
|
2019-04-16 00:06:45 +02:00
|
|
|
result = await hass.config_entries.flow.async_init(
|
2020-05-14 10:49:27 +02:00
|
|
|
AXIS_DOMAIN,
|
2019-04-16 00:06:45 +02:00
|
|
|
data={
|
2020-05-14 10:49:27 +02:00
|
|
|
CONF_HOST: "1.2.3.4",
|
|
|
|
CONF_PORT: 80,
|
2019-10-16 20:45:03 +02:00
|
|
|
"hostname": "name",
|
2020-01-05 10:11:17 +01:00
|
|
|
"properties": {"macaddress": MAC},
|
2019-04-16 00:06:45 +02:00
|
|
|
},
|
2019-07-31 12:25:30 -07:00
|
|
|
context={"source": "zeroconf"},
|
2019-04-16 00:06:45 +02:00
|
|
|
)
|
2019-03-24 16:16:50 +01:00
|
|
|
|
2019-07-31 12:25:30 -07:00
|
|
|
assert result["type"] == "form"
|
|
|
|
assert result["step_id"] == "user"
|
2019-03-24 16:16:50 +01:00
|
|
|
|
2020-01-05 10:11:17 +01:00
|
|
|
with patch("axis.AxisDevice") as mock_device:
|
|
|
|
|
|
|
|
setup_mock_axis_device(mock_device)
|
|
|
|
|
|
|
|
result = await hass.config_entries.flow.async_configure(
|
|
|
|
result["flow_id"],
|
|
|
|
user_input={
|
2020-05-14 10:49:27 +02:00
|
|
|
CONF_HOST: "1.2.3.4",
|
|
|
|
CONF_USERNAME: "user",
|
|
|
|
CONF_PASSWORD: "pass",
|
|
|
|
CONF_PORT: 80,
|
2020-01-05 10:11:17 +01:00
|
|
|
},
|
|
|
|
)
|
|
|
|
|
|
|
|
assert result["type"] == "create_entry"
|
|
|
|
assert result["title"] == f"prodnbr - {MAC}"
|
|
|
|
assert result["data"] == {
|
2020-05-14 10:49:27 +02:00
|
|
|
CONF_HOST: "1.2.3.4",
|
|
|
|
CONF_USERNAME: "user",
|
|
|
|
CONF_PASSWORD: "pass",
|
|
|
|
CONF_PORT: 80,
|
|
|
|
CONF_MAC: MAC,
|
|
|
|
CONF_MODEL: "prodnbr",
|
|
|
|
CONF_NAME: "prodnbr 0",
|
2020-01-05 10:11:17 +01:00
|
|
|
}
|
|
|
|
|
2020-05-14 10:49:27 +02:00
|
|
|
assert result["data"][CONF_NAME] == "prodnbr 0"
|
2020-01-05 10:11:17 +01:00
|
|
|
|
2019-03-24 16:16:50 +01:00
|
|
|
|
2019-05-22 00:36:26 +02:00
|
|
|
async def test_zeroconf_flow_already_configured(hass):
|
|
|
|
"""Test that zeroconf doesn't setup already configured devices."""
|
2020-01-04 08:58:18 +01:00
|
|
|
device = await setup_axis_integration(hass)
|
|
|
|
assert device.host == "1.2.3.4"
|
2019-03-24 16:16:50 +01:00
|
|
|
|
2019-04-16 00:06:45 +02:00
|
|
|
result = await hass.config_entries.flow.async_init(
|
2020-05-14 10:49:27 +02:00
|
|
|
AXIS_DOMAIN,
|
2019-04-16 00:06:45 +02:00
|
|
|
data={
|
2020-05-14 10:49:27 +02:00
|
|
|
CONF_HOST: "1.2.3.4",
|
|
|
|
CONF_PORT: 80,
|
2019-07-31 12:25:30 -07:00
|
|
|
"hostname": "name",
|
2020-01-05 10:11:17 +01:00
|
|
|
"properties": {"macaddress": MAC},
|
2019-04-16 00:06:45 +02:00
|
|
|
},
|
2019-07-31 12:25:30 -07:00
|
|
|
context={"source": "zeroconf"},
|
2019-04-16 00:06:45 +02:00
|
|
|
)
|
|
|
|
|
2019-07-31 12:25:30 -07:00
|
|
|
assert result["type"] == "abort"
|
|
|
|
assert result["reason"] == "already_configured"
|
2020-01-04 08:58:18 +01:00
|
|
|
assert device.host == "1.2.3.4"
|
|
|
|
|
|
|
|
|
|
|
|
async def test_zeroconf_flow_updated_configuration(hass):
|
|
|
|
"""Test that zeroconf update configuration with new parameters."""
|
|
|
|
device = await setup_axis_integration(hass)
|
|
|
|
assert device.host == "1.2.3.4"
|
2020-01-30 22:20:30 +01:00
|
|
|
assert device.config_entry.data == {
|
2020-05-14 10:49:27 +02:00
|
|
|
CONF_HOST: "1.2.3.4",
|
|
|
|
CONF_PORT: 80,
|
|
|
|
CONF_USERNAME: "username",
|
|
|
|
CONF_PASSWORD: "password",
|
|
|
|
CONF_MAC: MAC,
|
|
|
|
CONF_MODEL: MODEL,
|
|
|
|
CONF_NAME: NAME,
|
2020-01-30 22:20:30 +01:00
|
|
|
}
|
2020-01-04 08:58:18 +01:00
|
|
|
|
|
|
|
result = await hass.config_entries.flow.async_init(
|
2020-05-14 10:49:27 +02:00
|
|
|
AXIS_DOMAIN,
|
2020-01-04 08:58:18 +01:00
|
|
|
data={
|
2020-05-14 10:49:27 +02:00
|
|
|
CONF_HOST: "2.3.4.5",
|
|
|
|
CONF_PORT: 8080,
|
2020-01-04 08:58:18 +01:00
|
|
|
"hostname": "name",
|
|
|
|
"properties": {"macaddress": MAC},
|
|
|
|
},
|
|
|
|
context={"source": "zeroconf"},
|
|
|
|
)
|
|
|
|
|
|
|
|
assert result["type"] == "abort"
|
2020-01-30 22:20:30 +01:00
|
|
|
assert result["reason"] == "already_configured"
|
|
|
|
assert device.config_entry.data == {
|
2020-05-14 10:49:27 +02:00
|
|
|
CONF_HOST: "2.3.4.5",
|
|
|
|
CONF_PORT: 8080,
|
|
|
|
CONF_USERNAME: "username",
|
|
|
|
CONF_PASSWORD: "password",
|
|
|
|
CONF_MAC: MAC,
|
|
|
|
CONF_MODEL: MODEL,
|
|
|
|
CONF_NAME: NAME,
|
2020-01-30 22:20:30 +01:00
|
|
|
}
|
2019-03-24 16:16:50 +01:00
|
|
|
|
|
|
|
|
2019-06-10 18:12:17 +02: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(
|
2020-05-14 10:49:27 +02:00
|
|
|
AXIS_DOMAIN,
|
|
|
|
data={CONF_HOST: "169.254.3.4", "properties": {"macaddress": "01234567890"}},
|
2019-07-31 12:25:30 -07:00
|
|
|
context={"source": "zeroconf"},
|
2019-06-10 18:12:17 +02:00
|
|
|
)
|
|
|
|
|
2019-07-31 12:25:30 -07:00
|
|
|
assert result["type"] == "abort"
|
|
|
|
assert result["reason"] == "not_axis_device"
|
2019-06-10 18:12:17 +02:00
|
|
|
|
|
|
|
|
2019-05-22 00:36:26 +02:00
|
|
|
async def test_zeroconf_flow_ignore_link_local_address(hass):
|
|
|
|
"""Test that zeroconf doesn't setup devices with link local addresses."""
|
2019-04-16 00:06:45 +02:00
|
|
|
result = await hass.config_entries.flow.async_init(
|
2020-05-14 10:49:27 +02:00
|
|
|
AXIS_DOMAIN,
|
|
|
|
data={CONF_HOST: "169.254.3.4", "properties": {"macaddress": MAC}},
|
2019-07-31 12:25:30 -07:00
|
|
|
context={"source": "zeroconf"},
|
2019-04-16 00:06:45 +02:00
|
|
|
)
|
2019-03-24 16:16:50 +01:00
|
|
|
|
2019-07-31 12:25:30 -07:00
|
|
|
assert result["type"] == "abort"
|
|
|
|
assert result["reason"] == "link_local_address"
|