2018-11-27 20:21:25 +00:00
|
|
|
"""Tests for ZHA config flow."""
|
2019-12-10 05:00:04 +00:00
|
|
|
from unittest import mock
|
|
|
|
|
|
|
|
import asynctest
|
2019-10-21 23:30:56 +00:00
|
|
|
|
2018-11-27 20:21:25 +00:00
|
|
|
from homeassistant.components.zha import config_flow
|
2019-12-10 05:00:04 +00:00
|
|
|
from homeassistant.components.zha.core.const import CONTROLLER, DOMAIN, ZHA_GW_RADIO
|
|
|
|
import homeassistant.components.zha.core.registries
|
2019-10-21 23:30:56 +00:00
|
|
|
|
2018-11-27 20:21:25 +00:00
|
|
|
from tests.common import MockConfigEntry
|
|
|
|
|
|
|
|
|
|
|
|
async def test_user_flow(hass):
|
|
|
|
"""Test that config flow works."""
|
|
|
|
flow = config_flow.ZhaFlowHandler()
|
|
|
|
flow.hass = hass
|
|
|
|
|
2019-12-10 05:00:04 +00:00
|
|
|
with asynctest.patch(
|
2020-01-02 19:17:10 +00:00
|
|
|
"homeassistant.components.zha.config_flow.check_zigpy_connection",
|
2019-07-31 19:25:30 +00:00
|
|
|
return_value=False,
|
|
|
|
):
|
2018-11-27 20:21:25 +00:00
|
|
|
result = await flow.async_step_user(
|
2019-07-31 19:25:30 +00:00
|
|
|
user_input={"usb_path": "/dev/ttyUSB1", "radio_type": "ezsp"}
|
|
|
|
)
|
2018-11-27 20:21:25 +00:00
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
assert result["errors"] == {"base": "cannot_connect"}
|
2018-11-27 20:21:25 +00:00
|
|
|
|
2019-12-10 05:00:04 +00:00
|
|
|
with asynctest.patch(
|
2020-01-02 19:17:10 +00:00
|
|
|
"homeassistant.components.zha.config_flow.check_zigpy_connection",
|
2019-07-31 19:25:30 +00:00
|
|
|
return_value=True,
|
|
|
|
):
|
2018-11-27 20:21:25 +00:00
|
|
|
result = await flow.async_step_user(
|
2019-07-31 19:25:30 +00:00
|
|
|
user_input={"usb_path": "/dev/ttyUSB1", "radio_type": "ezsp"}
|
|
|
|
)
|
2018-11-27 20:21:25 +00:00
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
assert result["type"] == "create_entry"
|
|
|
|
assert result["title"] == "/dev/ttyUSB1"
|
|
|
|
assert result["data"] == {"usb_path": "/dev/ttyUSB1", "radio_type": "ezsp"}
|
2018-11-27 20:21:25 +00:00
|
|
|
|
|
|
|
|
|
|
|
async def test_user_flow_existing_config_entry(hass):
|
|
|
|
"""Test if config entry already exists."""
|
2019-07-31 19:25:30 +00:00
|
|
|
MockConfigEntry(domain=DOMAIN, data={"usb_path": "/dev/ttyUSB1"}).add_to_hass(hass)
|
2018-11-27 20:21:25 +00:00
|
|
|
flow = config_flow.ZhaFlowHandler()
|
|
|
|
flow.hass = hass
|
|
|
|
|
|
|
|
result = await flow.async_step_user()
|
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
assert result["type"] == "abort"
|
2018-11-27 20:21:25 +00:00
|
|
|
|
|
|
|
|
|
|
|
async def test_import_flow(hass):
|
|
|
|
"""Test import from configuration.yaml ."""
|
|
|
|
flow = config_flow.ZhaFlowHandler()
|
|
|
|
flow.hass = hass
|
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
result = await flow.async_step_import(
|
|
|
|
{"usb_path": "/dev/ttyUSB1", "radio_type": "xbee"}
|
|
|
|
)
|
2018-11-27 20:21:25 +00:00
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
assert result["type"] == "create_entry"
|
|
|
|
assert result["title"] == "/dev/ttyUSB1"
|
|
|
|
assert result["data"] == {"usb_path": "/dev/ttyUSB1", "radio_type": "xbee"}
|
2018-11-27 20:21:25 +00:00
|
|
|
|
|
|
|
|
|
|
|
async def test_import_flow_existing_config_entry(hass):
|
|
|
|
"""Test import from configuration.yaml ."""
|
2019-07-31 19:25:30 +00:00
|
|
|
MockConfigEntry(domain=DOMAIN, data={"usb_path": "/dev/ttyUSB1"}).add_to_hass(hass)
|
2018-11-27 20:21:25 +00:00
|
|
|
flow = config_flow.ZhaFlowHandler()
|
|
|
|
flow.hass = hass
|
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
result = await flow.async_step_import(
|
|
|
|
{"usb_path": "/dev/ttyUSB1", "radio_type": "xbee"}
|
|
|
|
)
|
2018-11-27 20:21:25 +00:00
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
assert result["type"] == "abort"
|
2019-12-10 05:00:04 +00:00
|
|
|
|
|
|
|
|
|
|
|
async def test_check_zigpy_connection():
|
|
|
|
"""Test config flow validator."""
|
|
|
|
|
|
|
|
mock_radio = asynctest.MagicMock()
|
|
|
|
mock_radio.connect = asynctest.CoroutineMock()
|
|
|
|
radio_cls = asynctest.MagicMock(return_value=mock_radio)
|
|
|
|
|
|
|
|
bad_radio = asynctest.MagicMock()
|
|
|
|
bad_radio.connect = asynctest.CoroutineMock(side_effect=Exception)
|
|
|
|
bad_radio_cls = asynctest.MagicMock(return_value=bad_radio)
|
|
|
|
|
|
|
|
mock_ctrl = asynctest.MagicMock()
|
|
|
|
mock_ctrl.startup = asynctest.CoroutineMock()
|
|
|
|
mock_ctrl.shutdown = asynctest.CoroutineMock()
|
|
|
|
ctrl_cls = asynctest.MagicMock(return_value=mock_ctrl)
|
|
|
|
new_radios = {
|
|
|
|
mock.sentinel.radio: {ZHA_GW_RADIO: radio_cls, CONTROLLER: ctrl_cls},
|
|
|
|
mock.sentinel.bad_radio: {ZHA_GW_RADIO: bad_radio_cls, CONTROLLER: ctrl_cls},
|
|
|
|
}
|
|
|
|
|
|
|
|
with mock.patch.dict(
|
|
|
|
homeassistant.components.zha.core.registries.RADIO_TYPES, new_radios, clear=True
|
|
|
|
):
|
|
|
|
assert not await config_flow.check_zigpy_connection(
|
|
|
|
mock.sentinel.usb_path, mock.sentinel.unk_radio, mock.sentinel.zigbee_db
|
|
|
|
)
|
|
|
|
assert mock_radio.connect.call_count == 0
|
|
|
|
assert bad_radio.connect.call_count == 0
|
|
|
|
assert mock_ctrl.startup.call_count == 0
|
|
|
|
assert mock_ctrl.shutdown.call_count == 0
|
|
|
|
|
|
|
|
# unsuccessful radio connect
|
|
|
|
assert not await config_flow.check_zigpy_connection(
|
|
|
|
mock.sentinel.usb_path, mock.sentinel.bad_radio, mock.sentinel.zigbee_db
|
|
|
|
)
|
|
|
|
assert mock_radio.connect.call_count == 0
|
|
|
|
assert bad_radio.connect.call_count == 1
|
|
|
|
assert mock_ctrl.startup.call_count == 0
|
|
|
|
assert mock_ctrl.shutdown.call_count == 0
|
|
|
|
|
|
|
|
# successful radio connect
|
|
|
|
assert await config_flow.check_zigpy_connection(
|
|
|
|
mock.sentinel.usb_path, mock.sentinel.radio, mock.sentinel.zigbee_db
|
|
|
|
)
|
|
|
|
assert mock_radio.connect.call_count == 1
|
|
|
|
assert bad_radio.connect.call_count == 1
|
|
|
|
assert mock_ctrl.startup.call_count == 1
|
|
|
|
assert mock_ctrl.shutdown.call_count == 1
|