hass-core/tests/components/lektrico/test_config_flow.py
Lektri.co 5bd736029f
Add lektrico integration (#102371)
* Add Lektrico Integration

* Make the changes proposed by Lash-L: new coordinator.py, new entity.py; use: translation_key, last_update_sucess, PlatformNotReady; remove: global variables

* Replace FlowResult with ConfigFlowResult and add tests.

* Remove unused lines.

* Remove Options from condif_flow

* Fix ruff and mypy.

* Fix CODEOWNERS.

* Run python3 -m script.hassfest.

* Correct rebase mistake.

* Make modifications suggested by emontnemery.

* Add pytest fixtures.

* Remove meaningless patches.

* Update .coveragerc

* Replace CONF_FRIENDLY_NAME with CONF_NAME.

* Remove underscores.

* Update tests.

* Update test file with is and no config_entries. .

* Set serial_number in DeviceInfo and add return type of the async_update_data to DataUpdateCoordinator.

* Use suggested_unit_of_measurement for KILO_WATT and replace Any in value_fn (sensor file).

* Add device class duration to charging_time sensor.

* Change raising  PlatformNotReady to raising IntegrationError.

* Test the unique id of the entry.

* Rename PF Lx with Power factor Lx and remove PF from strings.json.

* Remove comment.

* Make state and limit reason sensors to be enum sensors.

* Use result variable to check unique_id in test.

* Remove CONF_NAME from entry and __init__ from LektricoFlowHandler.

* Remove session parameter from LektricoDeviceDataUpdateCoordinator.

* Use config_entry: ConfigEntry in coordinator.

* Replace Connected,NeedAuth with Waiting for Authentication.

* Use lektricowifi 0.0.29.

* Use lektricowifi 0.0.39

* Use lektricowifi 0.0.40

* Use lektricowifi 0.0.41

* Replace hass.data with entry.runtime_data

* Delete .coveragerc

* Restructure the user step

* Fix tests

* Add returned value of _async_update_data to class DataUpdateCoordinator

* Use hw_version at DeviceInfo

* Remove a variable

* Use StateType

* Replace friendly_name with device_name

* Use sentence case in translation strings

* Uncomment and fix test_discovered_zeroconf

* Add type LektricoConfigEntry

* Remove commented code

* Remove the type of coordinator in sensor async_setup_entry

* Make zeroconf test end in ABORT, not FORM

* Remove all async_block_till_done from tests

* End test_user_setup_device_offline with CREATE_ENTRY

* Patch the full Device

* Add snapshot tests

* Overwrite the type LektricoSensorEntityDescription outside of the constructor

* Test separate already_configured for zeroconf

---------

Co-authored-by: mihaela.tarjoianu <mihaela.tarjoianu@scada.ro>
Co-authored-by: Erik Montnemery <erik@montnemery.com>
2024-08-30 13:20:15 +02:00

173 lines
5.5 KiB
Python

"""Tests for the Lektrico Charging Station config flow."""
import dataclasses
from ipaddress import ip_address
from lektricowifi import DeviceConnectionError
from homeassistant.components.lektrico.const import DOMAIN
from homeassistant.config_entries import SOURCE_USER, SOURCE_ZEROCONF
from homeassistant.const import (
ATTR_HW_VERSION,
ATTR_SERIAL_NUMBER,
CONF_HOST,
CONF_TYPE,
)
from homeassistant.core import HomeAssistant
from homeassistant.data_entry_flow import FlowResultType
from .conftest import (
MOCKED_DEVICE_BOARD_REV,
MOCKED_DEVICE_IP_ADDRESS,
MOCKED_DEVICE_SERIAL_NUMBER,
MOCKED_DEVICE_TYPE,
MOCKED_DEVICE_ZEROCONF_DATA,
)
from tests.common import MockConfigEntry
async def test_user_setup(hass: HomeAssistant, mock_device, mock_setup_entry) -> None:
"""Test manually setting up."""
result = await hass.config_entries.flow.async_init(
DOMAIN,
context={"source": SOURCE_USER},
)
assert result["type"] is FlowResultType.FORM
assert result["step_id"] == SOURCE_USER
assert "flow_id" in result
result = await hass.config_entries.flow.async_configure(
result["flow_id"],
{
CONF_HOST: MOCKED_DEVICE_IP_ADDRESS,
},
)
assert result.get("type") is FlowResultType.CREATE_ENTRY
assert result.get("title") == f"{MOCKED_DEVICE_TYPE}_{MOCKED_DEVICE_SERIAL_NUMBER}"
assert result.get("data") == {
CONF_HOST: MOCKED_DEVICE_IP_ADDRESS,
ATTR_SERIAL_NUMBER: MOCKED_DEVICE_SERIAL_NUMBER,
CONF_TYPE: MOCKED_DEVICE_TYPE,
ATTR_HW_VERSION: MOCKED_DEVICE_BOARD_REV,
}
assert "result" in result
assert len(mock_setup_entry.mock_calls) == 1
assert result.get("result").unique_id == MOCKED_DEVICE_SERIAL_NUMBER
async def test_user_setup_already_exists(
hass: HomeAssistant, mock_device, mock_config_entry: MockConfigEntry
) -> None:
"""Test manually setting up when the device already exists."""
mock_config_entry.add_to_hass(hass)
result = await hass.config_entries.flow.async_init(
DOMAIN, context={"source": SOURCE_USER}
)
assert result["type"] is FlowResultType.FORM
assert result["step_id"] == "user"
assert not result["errors"]
result = await hass.config_entries.flow.async_configure(
result["flow_id"],
{
CONF_HOST: MOCKED_DEVICE_IP_ADDRESS,
},
)
assert result["type"] is FlowResultType.ABORT
assert result["reason"] == "already_configured"
async def test_user_setup_device_offline(hass: HomeAssistant, mock_device) -> None:
"""Test manually setting up when device is offline."""
result = await hass.config_entries.flow.async_init(
DOMAIN, context={"source": SOURCE_USER}
)
assert result["type"] is FlowResultType.FORM
assert result["step_id"] == "user"
assert not result["errors"]
mock_device.device_config.side_effect = DeviceConnectionError
result = await hass.config_entries.flow.async_configure(
result["flow_id"],
{
CONF_HOST: MOCKED_DEVICE_IP_ADDRESS,
},
)
assert result["type"] is FlowResultType.FORM
assert result["errors"] == {CONF_HOST: "cannot_connect"}
assert result["step_id"] == "user"
mock_device.device_config.side_effect = None
result = await hass.config_entries.flow.async_configure(
result["flow_id"],
{
CONF_HOST: MOCKED_DEVICE_IP_ADDRESS,
},
)
assert result["type"] is FlowResultType.CREATE_ENTRY
async def test_discovered_zeroconf(
hass: HomeAssistant, mock_device, mock_setup_entry
) -> None:
"""Test we can setup when discovered from zeroconf."""
result = await hass.config_entries.flow.async_init(
DOMAIN,
context={"source": SOURCE_ZEROCONF},
data=MOCKED_DEVICE_ZEROCONF_DATA,
)
assert result["type"] is FlowResultType.FORM
assert result["errors"] is None
assert result.get("step_id") == "confirm"
result2 = await hass.config_entries.flow.async_configure(result["flow_id"], {})
assert result2["type"] is FlowResultType.CREATE_ENTRY
assert result2["data"] == {
CONF_HOST: MOCKED_DEVICE_IP_ADDRESS,
ATTR_SERIAL_NUMBER: MOCKED_DEVICE_SERIAL_NUMBER,
CONF_TYPE: MOCKED_DEVICE_TYPE,
ATTR_HW_VERSION: MOCKED_DEVICE_BOARD_REV,
}
assert result2["title"] == f"{MOCKED_DEVICE_TYPE}_{MOCKED_DEVICE_SERIAL_NUMBER}"
async def test_zeroconf_setup_already_exists(
hass: HomeAssistant, mock_device, mock_config_entry: MockConfigEntry
) -> None:
"""Test we abort zeroconf flow if device already configured."""
mock_config_entry.add_to_hass(hass)
zc_data_new_ip = dataclasses.replace(MOCKED_DEVICE_ZEROCONF_DATA)
zc_data_new_ip.ip_address = ip_address(MOCKED_DEVICE_IP_ADDRESS)
result = await hass.config_entries.flow.async_init(
DOMAIN,
context={"source": SOURCE_ZEROCONF},
data=zc_data_new_ip,
)
assert result["type"] is FlowResultType.ABORT
assert result["reason"] == "already_configured"
async def test_discovered_zeroconf_device_connection_error(
hass: HomeAssistant, mock_device
) -> None:
"""Test we can setup when discovered from zeroconf but device went offline."""
mock_device.device_config.side_effect = DeviceConnectionError
result = await hass.config_entries.flow.async_init(
DOMAIN,
context={"source": SOURCE_ZEROCONF},
data=MOCKED_DEVICE_ZEROCONF_DATA,
)
assert result["type"] is FlowResultType.ABORT
assert result["reason"] == "cannot_connect"