Add SenseME integration (#62909)
Co-authored-by: Big Mike <mikelawrence@users.noreply.github.com>
This commit is contained in:
parent
509ddc84a5
commit
943aaaeb3f
18 changed files with 929 additions and 0 deletions
|
@ -949,6 +949,10 @@ omit =
|
||||||
homeassistant/components/sense/sensor.py
|
homeassistant/components/sense/sensor.py
|
||||||
homeassistant/components/sensehat/light.py
|
homeassistant/components/sensehat/light.py
|
||||||
homeassistant/components/sensehat/sensor.py
|
homeassistant/components/sensehat/sensor.py
|
||||||
|
homeassistant/components/senseme/__init__.py
|
||||||
|
homeassistant/components/senseme/discovery.py
|
||||||
|
homeassistant/components/senseme/entity.py
|
||||||
|
homeassistant/components/senseme/fan.py
|
||||||
homeassistant/components/sensibo/__init__.py
|
homeassistant/components/sensibo/__init__.py
|
||||||
homeassistant/components/sensibo/climate.py
|
homeassistant/components/sensibo/climate.py
|
||||||
homeassistant/components/serial/sensor.py
|
homeassistant/components/serial/sensor.py
|
||||||
|
|
|
@ -123,6 +123,7 @@ homeassistant.components.samsungtv.*
|
||||||
homeassistant.components.scene.*
|
homeassistant.components.scene.*
|
||||||
homeassistant.components.select.*
|
homeassistant.components.select.*
|
||||||
homeassistant.components.sensor.*
|
homeassistant.components.sensor.*
|
||||||
|
homeassistant.components.senseme.*
|
||||||
homeassistant.components.shelly.*
|
homeassistant.components.shelly.*
|
||||||
homeassistant.components.simplisafe.*
|
homeassistant.components.simplisafe.*
|
||||||
homeassistant.components.slack.*
|
homeassistant.components.slack.*
|
||||||
|
|
|
@ -801,6 +801,8 @@ homeassistant/components/select/* @home-assistant/core
|
||||||
tests/components/select/* @home-assistant/core
|
tests/components/select/* @home-assistant/core
|
||||||
homeassistant/components/sense/* @kbickar
|
homeassistant/components/sense/* @kbickar
|
||||||
tests/components/sense/* @kbickar
|
tests/components/sense/* @kbickar
|
||||||
|
homeassistant/components/senseme/* @mikelawrence @bdraco
|
||||||
|
tests/components/senseme/* @mikelawrence @bdraco
|
||||||
homeassistant/components/sensibo/* @andrey-git @gjohansson-ST
|
homeassistant/components/sensibo/* @andrey-git @gjohansson-ST
|
||||||
tests/components/sensibo/* @andrey-git @gjohansson-ST
|
tests/components/sensibo/* @andrey-git @gjohansson-ST
|
||||||
homeassistant/components/sentry/* @dcramer @frenck
|
homeassistant/components/sentry/* @dcramer @frenck
|
||||||
|
|
36
homeassistant/components/senseme/__init__.py
Normal file
36
homeassistant/components/senseme/__init__.py
Normal file
|
@ -0,0 +1,36 @@
|
||||||
|
"""The SenseME integration."""
|
||||||
|
from __future__ import annotations
|
||||||
|
|
||||||
|
from aiosenseme import async_get_device_by_device_info
|
||||||
|
|
||||||
|
from homeassistant.config_entries import ConfigEntry
|
||||||
|
from homeassistant.core import HomeAssistant
|
||||||
|
from homeassistant.exceptions import ConfigEntryNotReady
|
||||||
|
|
||||||
|
from .const import CONF_INFO, DOMAIN, PLATFORMS, UPDATE_RATE
|
||||||
|
from .discovery import async_start_discovery
|
||||||
|
|
||||||
|
|
||||||
|
async def async_setup_entry(hass: HomeAssistant, entry: ConfigEntry) -> bool:
|
||||||
|
"""Set up SenseME from a config entry."""
|
||||||
|
async_start_discovery(hass)
|
||||||
|
|
||||||
|
status, device = await async_get_device_by_device_info(
|
||||||
|
info=entry.data[CONF_INFO], start_first=True, refresh_minutes=UPDATE_RATE
|
||||||
|
)
|
||||||
|
if not status:
|
||||||
|
device.stop()
|
||||||
|
raise ConfigEntryNotReady(f"Connect to address {device.address} failed")
|
||||||
|
|
||||||
|
await device.async_update(not status)
|
||||||
|
|
||||||
|
hass.data[DOMAIN][entry.entry_id] = device
|
||||||
|
hass.config_entries.async_setup_platforms(entry, PLATFORMS)
|
||||||
|
|
||||||
|
return True
|
||||||
|
|
||||||
|
|
||||||
|
async def async_unload_entry(hass: HomeAssistant, entry: ConfigEntry) -> bool:
|
||||||
|
"""Unload a config entry."""
|
||||||
|
hass.data[DOMAIN][entry.entry_id].stop()
|
||||||
|
return await hass.config_entries.async_unload_platforms(entry, PLATFORMS)
|
135
homeassistant/components/senseme/config_flow.py
Normal file
135
homeassistant/components/senseme/config_flow.py
Normal file
|
@ -0,0 +1,135 @@
|
||||||
|
"""Config flow for SenseME."""
|
||||||
|
from __future__ import annotations
|
||||||
|
|
||||||
|
import ipaddress
|
||||||
|
from typing import Any
|
||||||
|
|
||||||
|
from aiosenseme import SensemeDevice, async_get_device_by_ip_address
|
||||||
|
import voluptuous as vol
|
||||||
|
|
||||||
|
from homeassistant import config_entries
|
||||||
|
from homeassistant.const import CONF_DEVICE, CONF_HOST, CONF_ID
|
||||||
|
from homeassistant.data_entry_flow import FlowResult
|
||||||
|
from homeassistant.helpers.typing import DiscoveryInfoType
|
||||||
|
|
||||||
|
from .const import CONF_HOST_MANUAL, CONF_INFO, DOMAIN
|
||||||
|
from .discovery import async_discover, async_get_discovered_device
|
||||||
|
|
||||||
|
DISCOVER_TIMEOUT = 5
|
||||||
|
|
||||||
|
|
||||||
|
class SensemeFlowHandler(config_entries.ConfigFlow, domain=DOMAIN):
|
||||||
|
"""Handle SenseME discovery config flow."""
|
||||||
|
|
||||||
|
VERSION = 1
|
||||||
|
|
||||||
|
def __init__(self) -> None:
|
||||||
|
"""Initialize the SenseME config flow."""
|
||||||
|
self._discovered_devices: list[SensemeDevice] | None = None
|
||||||
|
self._discovered_device: SensemeDevice | None = None
|
||||||
|
|
||||||
|
async def async_step_discovery(
|
||||||
|
self, discovery_info: DiscoveryInfoType
|
||||||
|
) -> FlowResult:
|
||||||
|
"""Handle discovery."""
|
||||||
|
uuid = discovery_info[CONF_ID]
|
||||||
|
device = async_get_discovered_device(self.hass, discovery_info[CONF_ID])
|
||||||
|
host = device.address
|
||||||
|
await self.async_set_unique_id(uuid)
|
||||||
|
for entry in self._async_current_entries(include_ignore=False):
|
||||||
|
if entry.data[CONF_INFO]["address"] == host:
|
||||||
|
return self.async_abort(reason="already_configured")
|
||||||
|
if entry.unique_id != uuid:
|
||||||
|
continue
|
||||||
|
if entry.data[CONF_INFO]["address"] != host:
|
||||||
|
self.hass.config_entries.async_update_entry(
|
||||||
|
entry, data={CONF_INFO: {**entry.data[CONF_INFO], "address": host}}
|
||||||
|
)
|
||||||
|
return self.async_abort(reason="already_configured")
|
||||||
|
self._discovered_device = device
|
||||||
|
return await self.async_step_discovery_confirm()
|
||||||
|
|
||||||
|
async def async_step_discovery_confirm(
|
||||||
|
self, user_input: dict[str, Any] | None = None
|
||||||
|
) -> FlowResult:
|
||||||
|
"""Confirm discovery."""
|
||||||
|
device = self._discovered_device
|
||||||
|
assert device is not None
|
||||||
|
|
||||||
|
if user_input is not None:
|
||||||
|
return await self._async_entry_for_device(device)
|
||||||
|
placeholders = {
|
||||||
|
"name": device.name,
|
||||||
|
"model": device.model,
|
||||||
|
"host": device.address,
|
||||||
|
}
|
||||||
|
self.context["title_placeholders"] = placeholders
|
||||||
|
return self.async_show_form(
|
||||||
|
step_id="discovery_confirm", description_placeholders=placeholders
|
||||||
|
)
|
||||||
|
|
||||||
|
async def _async_entry_for_device(self, device: SensemeDevice) -> FlowResult:
|
||||||
|
"""Create a config entry for a device."""
|
||||||
|
await self.async_set_unique_id(device.uuid, raise_on_progress=False)
|
||||||
|
self._abort_if_unique_id_configured()
|
||||||
|
return self.async_create_entry(
|
||||||
|
title=device.name,
|
||||||
|
data={CONF_INFO: device.get_device_info},
|
||||||
|
)
|
||||||
|
|
||||||
|
async def async_step_manual(
|
||||||
|
self, user_input: dict[str, Any] | None = None
|
||||||
|
) -> FlowResult:
|
||||||
|
"""Handle manual entry of an ip address."""
|
||||||
|
errors = {}
|
||||||
|
if user_input is not None:
|
||||||
|
host = user_input[CONF_HOST]
|
||||||
|
try:
|
||||||
|
ipaddress.ip_address(host)
|
||||||
|
except ValueError:
|
||||||
|
errors[CONF_HOST] = "invalid_host"
|
||||||
|
else:
|
||||||
|
if device := await async_get_device_by_ip_address(host):
|
||||||
|
device.stop()
|
||||||
|
return await self._async_entry_for_device(device)
|
||||||
|
|
||||||
|
errors[CONF_HOST] = "cannot_connect"
|
||||||
|
|
||||||
|
return self.async_show_form(
|
||||||
|
step_id="manual",
|
||||||
|
data_schema=vol.Schema({vol.Required(CONF_HOST): str}),
|
||||||
|
errors=errors,
|
||||||
|
)
|
||||||
|
|
||||||
|
async def async_step_user(
|
||||||
|
self, user_input: dict[str, Any] | None = None
|
||||||
|
) -> FlowResult:
|
||||||
|
"""Handle a flow initialized by the user."""
|
||||||
|
if self._discovered_devices is None:
|
||||||
|
self._discovered_devices = await async_discover(self.hass, DISCOVER_TIMEOUT)
|
||||||
|
current_ids = self._async_current_ids()
|
||||||
|
device_selection = {
|
||||||
|
device.uuid: device.name
|
||||||
|
for device in self._discovered_devices
|
||||||
|
if device.uuid not in current_ids
|
||||||
|
}
|
||||||
|
|
||||||
|
if not device_selection:
|
||||||
|
return await self.async_step_manual(user_input=None)
|
||||||
|
|
||||||
|
device_selection[None] = CONF_HOST_MANUAL
|
||||||
|
|
||||||
|
if user_input is not None:
|
||||||
|
if user_input[CONF_DEVICE] is None:
|
||||||
|
return await self.async_step_manual()
|
||||||
|
|
||||||
|
for device in self._discovered_devices:
|
||||||
|
if device.uuid == user_input[CONF_DEVICE]:
|
||||||
|
return await self._async_entry_for_device(device)
|
||||||
|
|
||||||
|
return self.async_show_form(
|
||||||
|
step_id="user",
|
||||||
|
data_schema=vol.Schema(
|
||||||
|
{vol.Required(CONF_DEVICE): vol.In(device_selection)}
|
||||||
|
),
|
||||||
|
)
|
23
homeassistant/components/senseme/const.py
Normal file
23
homeassistant/components/senseme/const.py
Normal file
|
@ -0,0 +1,23 @@
|
||||||
|
"""Constants for the SenseME integration."""
|
||||||
|
|
||||||
|
|
||||||
|
from homeassistant.const import Platform
|
||||||
|
|
||||||
|
DOMAIN = "senseme"
|
||||||
|
|
||||||
|
# Periodic fan update rate in minutes
|
||||||
|
UPDATE_RATE = 1
|
||||||
|
|
||||||
|
# data storage
|
||||||
|
CONF_INFO = "info"
|
||||||
|
CONF_HOST_MANUAL = "IP Address"
|
||||||
|
DISCOVERY = "discovery"
|
||||||
|
|
||||||
|
# Fan Preset Modes
|
||||||
|
PRESET_MODE_WHOOSH = "Whoosh"
|
||||||
|
|
||||||
|
# Fan Directions
|
||||||
|
SENSEME_DIRECTION_FORWARD = "FWD"
|
||||||
|
SENSEME_DIRECTION_REVERSE = "REV"
|
||||||
|
|
||||||
|
PLATFORMS = [Platform.FAN]
|
63
homeassistant/components/senseme/discovery.py
Normal file
63
homeassistant/components/senseme/discovery.py
Normal file
|
@ -0,0 +1,63 @@
|
||||||
|
"""The SenseME integration discovery."""
|
||||||
|
from __future__ import annotations
|
||||||
|
|
||||||
|
import asyncio
|
||||||
|
|
||||||
|
from aiosenseme import SensemeDevice, SensemeDiscovery
|
||||||
|
|
||||||
|
from homeassistant import config_entries
|
||||||
|
from homeassistant.const import CONF_ID
|
||||||
|
from homeassistant.core import HomeAssistant, callback
|
||||||
|
|
||||||
|
from .const import DISCOVERY, DOMAIN
|
||||||
|
|
||||||
|
|
||||||
|
@callback
|
||||||
|
def async_start_discovery(hass: HomeAssistant) -> bool:
|
||||||
|
"""Start discovery if its not already running."""
|
||||||
|
domain_data = hass.data.setdefault(DOMAIN, {})
|
||||||
|
if DISCOVERY in domain_data:
|
||||||
|
return False # already running
|
||||||
|
discovery = domain_data[DISCOVERY] = SensemeDiscovery(False)
|
||||||
|
discovery.add_callback(lambda devices: async_trigger_discovery(hass, devices))
|
||||||
|
discovery.start()
|
||||||
|
return True # started
|
||||||
|
|
||||||
|
|
||||||
|
@callback
|
||||||
|
def async_get_discovered_device(hass: HomeAssistant, uuid: str) -> SensemeDevice:
|
||||||
|
"""Return a discovered device."""
|
||||||
|
discovery: SensemeDiscovery = hass.data[DOMAIN][DISCOVERY]
|
||||||
|
devices: list[SensemeDevice] = discovery.devices
|
||||||
|
for discovered_device in devices:
|
||||||
|
if discovered_device.uuid == uuid:
|
||||||
|
return discovered_device
|
||||||
|
raise RuntimeError("Discovered device unexpectedly disappeared")
|
||||||
|
|
||||||
|
|
||||||
|
async def async_discover(hass: HomeAssistant, timeout: float) -> list[SensemeDevice]:
|
||||||
|
"""Discover devices or restart it if its already running."""
|
||||||
|
started = async_start_discovery(hass)
|
||||||
|
discovery: SensemeDiscovery = hass.data[DOMAIN][DISCOVERY]
|
||||||
|
if not started: # already running
|
||||||
|
discovery.stop()
|
||||||
|
discovery.start()
|
||||||
|
await asyncio.sleep(timeout)
|
||||||
|
devices: list[SensemeDevice] = discovery.devices
|
||||||
|
return devices
|
||||||
|
|
||||||
|
|
||||||
|
@callback
|
||||||
|
def async_trigger_discovery(
|
||||||
|
hass: HomeAssistant,
|
||||||
|
discovered_devices: list[SensemeDevice],
|
||||||
|
) -> None:
|
||||||
|
"""Trigger config flows for discovered devices."""
|
||||||
|
for device in discovered_devices:
|
||||||
|
hass.async_create_task(
|
||||||
|
hass.config_entries.flow.async_init(
|
||||||
|
DOMAIN,
|
||||||
|
context={"source": config_entries.SOURCE_DISCOVERY},
|
||||||
|
data={CONF_ID: device.uuid},
|
||||||
|
)
|
||||||
|
)
|
54
homeassistant/components/senseme/entity.py
Normal file
54
homeassistant/components/senseme/entity.py
Normal file
|
@ -0,0 +1,54 @@
|
||||||
|
"""The SenseME integration entities."""
|
||||||
|
from __future__ import annotations
|
||||||
|
|
||||||
|
from aiosenseme import SensemeDevice
|
||||||
|
|
||||||
|
from homeassistant.core import callback
|
||||||
|
from homeassistant.helpers import device_registry as dr
|
||||||
|
from homeassistant.helpers.entity import DeviceInfo, Entity
|
||||||
|
|
||||||
|
|
||||||
|
class SensemeEntity(Entity):
|
||||||
|
"""Base class for senseme entities."""
|
||||||
|
|
||||||
|
_attr_should_poll = False
|
||||||
|
|
||||||
|
def __init__(self, device: SensemeDevice, name: str) -> None:
|
||||||
|
"""Initialize the entity."""
|
||||||
|
self._device = device
|
||||||
|
self._attr_name = name
|
||||||
|
self._attr_device_info = DeviceInfo(
|
||||||
|
connections={(dr.CONNECTION_NETWORK_MAC, self._device.mac)},
|
||||||
|
name=self._device.name,
|
||||||
|
manufacturer="Big Ass Fans",
|
||||||
|
model=self._device.model,
|
||||||
|
sw_version=self._device.fw_version,
|
||||||
|
suggested_area=self._device.room_name,
|
||||||
|
)
|
||||||
|
|
||||||
|
@property
|
||||||
|
def extra_state_attributes(self) -> dict:
|
||||||
|
"""Get the current device state attributes."""
|
||||||
|
return {
|
||||||
|
"room_name": self._device.room_name,
|
||||||
|
"room_type": self._device.room_type,
|
||||||
|
}
|
||||||
|
|
||||||
|
@callback
|
||||||
|
def _async_update_attrs(self) -> None:
|
||||||
|
"""Update attrs from device."""
|
||||||
|
self._attr_available = self._device.available
|
||||||
|
|
||||||
|
@callback
|
||||||
|
def _async_update_from_device(self) -> None:
|
||||||
|
"""Process an update from the device."""
|
||||||
|
self._async_update_attrs()
|
||||||
|
self.async_write_ha_state()
|
||||||
|
|
||||||
|
async def async_added_to_hass(self) -> None:
|
||||||
|
"""Add data updated listener after this object has been initialized."""
|
||||||
|
self._device.add_callback(self._async_update_from_device)
|
||||||
|
|
||||||
|
async def async_will_remove_from_hass(self) -> None:
|
||||||
|
"""Remove data updated listener after this object has been initialized."""
|
||||||
|
self._device.remove_callback(self._async_update_from_device)
|
125
homeassistant/components/senseme/fan.py
Normal file
125
homeassistant/components/senseme/fan.py
Normal file
|
@ -0,0 +1,125 @@
|
||||||
|
"""Support for Big Ass Fans SenseME fan."""
|
||||||
|
from __future__ import annotations
|
||||||
|
|
||||||
|
import math
|
||||||
|
from typing import Any
|
||||||
|
|
||||||
|
from aiosenseme import SensemeFan
|
||||||
|
|
||||||
|
from homeassistant import config_entries
|
||||||
|
from homeassistant.components.fan import (
|
||||||
|
DIRECTION_FORWARD,
|
||||||
|
DIRECTION_REVERSE,
|
||||||
|
SUPPORT_DIRECTION,
|
||||||
|
SUPPORT_SET_SPEED,
|
||||||
|
FanEntity,
|
||||||
|
)
|
||||||
|
from homeassistant.core import HomeAssistant, callback
|
||||||
|
from homeassistant.helpers.entity_platform import AddEntitiesCallback
|
||||||
|
from homeassistant.util.percentage import (
|
||||||
|
percentage_to_ranged_value,
|
||||||
|
ranged_value_to_percentage,
|
||||||
|
)
|
||||||
|
|
||||||
|
from .const import (
|
||||||
|
DOMAIN,
|
||||||
|
PRESET_MODE_WHOOSH,
|
||||||
|
SENSEME_DIRECTION_FORWARD,
|
||||||
|
SENSEME_DIRECTION_REVERSE,
|
||||||
|
)
|
||||||
|
from .entity import SensemeEntity
|
||||||
|
|
||||||
|
SENSEME_DIRECTION_TO_HASS = {
|
||||||
|
SENSEME_DIRECTION_FORWARD: DIRECTION_FORWARD,
|
||||||
|
SENSEME_DIRECTION_REVERSE: DIRECTION_REVERSE,
|
||||||
|
}
|
||||||
|
HASS_DIRECTION_TO_SENSEME = {v: k for k, v in SENSEME_DIRECTION_TO_HASS.items()}
|
||||||
|
|
||||||
|
|
||||||
|
async def async_setup_entry(
|
||||||
|
hass: HomeAssistant,
|
||||||
|
entry: config_entries.ConfigEntry,
|
||||||
|
async_add_entities: AddEntitiesCallback,
|
||||||
|
) -> None:
|
||||||
|
"""Set up SenseME fans."""
|
||||||
|
device = hass.data[DOMAIN][entry.entry_id]
|
||||||
|
if device.is_fan:
|
||||||
|
async_add_entities([HASensemeFan(device)])
|
||||||
|
|
||||||
|
|
||||||
|
class HASensemeFan(SensemeEntity, FanEntity):
|
||||||
|
"""SenseME ceiling fan component."""
|
||||||
|
|
||||||
|
_attr_supported_features = SUPPORT_SET_SPEED | SUPPORT_DIRECTION
|
||||||
|
_attr_preset_modes = [PRESET_MODE_WHOOSH]
|
||||||
|
|
||||||
|
def __init__(self, device: SensemeFan) -> None:
|
||||||
|
"""Initialize the entity."""
|
||||||
|
super().__init__(device, device.name)
|
||||||
|
self._attr_speed_count = self._device.fan_speed_max
|
||||||
|
self._attr_unique_id = f"{self._device.uuid}-FAN" # for legacy compat
|
||||||
|
self._async_update_attrs()
|
||||||
|
|
||||||
|
@callback
|
||||||
|
def _async_update_attrs(self) -> None:
|
||||||
|
"""Update attrs from device."""
|
||||||
|
self._attr_is_on = self._device.fan_on
|
||||||
|
self._attr_current_direction = SENSEME_DIRECTION_TO_HASS.get(
|
||||||
|
self._device.fan_dir, DIRECTION_FORWARD # None also means forward
|
||||||
|
)
|
||||||
|
if self._device.fan_speed is not None:
|
||||||
|
self._attr_percentage = ranged_value_to_percentage(
|
||||||
|
self._device.fan_speed_limits, self._device.fan_speed
|
||||||
|
)
|
||||||
|
else:
|
||||||
|
self._attr_percentage = None
|
||||||
|
whoosh = self._device.fan_whoosh_mode
|
||||||
|
self._attr_preset_mode = whoosh if whoosh else None
|
||||||
|
super()._async_update_attrs()
|
||||||
|
|
||||||
|
@property
|
||||||
|
def extra_state_attributes(self) -> dict:
|
||||||
|
"""Get the current device state attributes."""
|
||||||
|
return {
|
||||||
|
"auto_comfort": self._device.fan_autocomfort.capitalize(),
|
||||||
|
"smartmode": self._device.fan_smartmode.capitalize(),
|
||||||
|
**super().extra_state_attributes,
|
||||||
|
}
|
||||||
|
|
||||||
|
async def async_set_percentage(self, percentage: int) -> None:
|
||||||
|
"""Set the speed of the fan, as a percentage."""
|
||||||
|
self._device.fan_speed = math.ceil(
|
||||||
|
percentage_to_ranged_value(self._device.fan_speed_limits, percentage)
|
||||||
|
)
|
||||||
|
|
||||||
|
async def async_turn_on(
|
||||||
|
self,
|
||||||
|
speed: str | None = None,
|
||||||
|
percentage: int | None = None,
|
||||||
|
preset_mode: str | None = None,
|
||||||
|
**kwargs: Any,
|
||||||
|
) -> None:
|
||||||
|
"""Turn the fan on with a percentage or preset mode."""
|
||||||
|
if preset_mode is not None:
|
||||||
|
await self.async_set_preset_mode(preset_mode)
|
||||||
|
elif percentage is None:
|
||||||
|
self._device.fan_on = True
|
||||||
|
else:
|
||||||
|
await self.async_set_percentage(percentage)
|
||||||
|
|
||||||
|
async def async_turn_off(self, **kwargs: Any) -> None:
|
||||||
|
"""Turn the fan off."""
|
||||||
|
self._device.fan_on = False
|
||||||
|
|
||||||
|
async def async_set_preset_mode(self, preset_mode: str) -> None:
|
||||||
|
"""Set the preset mode of the fan."""
|
||||||
|
if preset_mode != PRESET_MODE_WHOOSH:
|
||||||
|
raise ValueError(f"Invalid preset mode: {preset_mode}")
|
||||||
|
# Sleep mode must be off for Whoosh to work.
|
||||||
|
if self._device.sleep_mode:
|
||||||
|
self._device.sleep_mode = False
|
||||||
|
self._device.fan_whoosh_mode = True
|
||||||
|
|
||||||
|
async def async_set_direction(self, direction: str) -> None:
|
||||||
|
"""Set the direction of the fan."""
|
||||||
|
self._device.fan_dir = HASS_DIRECTION_TO_SENSEME[direction]
|
13
homeassistant/components/senseme/manifest.json
Normal file
13
homeassistant/components/senseme/manifest.json
Normal file
|
@ -0,0 +1,13 @@
|
||||||
|
{
|
||||||
|
"domain": "senseme",
|
||||||
|
"name": "SenseME",
|
||||||
|
"config_flow": true,
|
||||||
|
"documentation": "https://www.home-assistant.io/integrations/senseme",
|
||||||
|
"requirements": [
|
||||||
|
"aiosenseme==0.5.5"
|
||||||
|
],
|
||||||
|
"codeowners": [
|
||||||
|
"@mikelawrence", "@bdraco"
|
||||||
|
],
|
||||||
|
"iot_class": "local_push"
|
||||||
|
}
|
29
homeassistant/components/senseme/strings.json
Normal file
29
homeassistant/components/senseme/strings.json
Normal file
|
@ -0,0 +1,29 @@
|
||||||
|
{
|
||||||
|
"config": {
|
||||||
|
"flow_title": "{name} - {model} ({host})",
|
||||||
|
"step": {
|
||||||
|
"user": {
|
||||||
|
"description": "Select a device, or choose 'IP Address' to manually enter an IP Address.",
|
||||||
|
"data": {
|
||||||
|
"device": "Device"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"discovery_confirm": {
|
||||||
|
"description": "Do you want to setup {name} - {model} ({host})?"
|
||||||
|
},
|
||||||
|
"manual": {
|
||||||
|
"description": "Enter an IP Address.",
|
||||||
|
"data": {
|
||||||
|
"host": "[%key:common::config_flow::data::host%]"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"abort": {
|
||||||
|
"already_configured": "[%key:common::config_flow::abort::already_configured_device%]"
|
||||||
|
},
|
||||||
|
"error": {
|
||||||
|
"invalid_host": "[%key:common::config_flow::error::invalid_host%]",
|
||||||
|
"cannot_connect": "[%key:common::config_flow::error::cannot_connect%]"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
29
homeassistant/components/senseme/translations/en.json
Normal file
29
homeassistant/components/senseme/translations/en.json
Normal file
|
@ -0,0 +1,29 @@
|
||||||
|
{
|
||||||
|
"config": {
|
||||||
|
"abort": {
|
||||||
|
"already_configured": "Device is already configured"
|
||||||
|
},
|
||||||
|
"error": {
|
||||||
|
"cannot_connect": "Failed to connect",
|
||||||
|
"invalid_host": "Invalid hostname or IP address"
|
||||||
|
},
|
||||||
|
"flow_title": "{name} - {model} ({host})",
|
||||||
|
"step": {
|
||||||
|
"discovery_confirm": {
|
||||||
|
"description": "Do you want to setup {name} - {model} ({host})?"
|
||||||
|
},
|
||||||
|
"manual": {
|
||||||
|
"data": {
|
||||||
|
"host": "Host"
|
||||||
|
},
|
||||||
|
"description": "Enter an IP Address."
|
||||||
|
},
|
||||||
|
"user": {
|
||||||
|
"data": {
|
||||||
|
"device": "Device"
|
||||||
|
},
|
||||||
|
"description": "Select a device, or choose 'IP Address' to manually enter an IP Address."
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
|
@ -269,6 +269,7 @@ FLOWS = [
|
||||||
"samsungtv",
|
"samsungtv",
|
||||||
"screenlogic",
|
"screenlogic",
|
||||||
"sense",
|
"sense",
|
||||||
|
"senseme",
|
||||||
"sensibo",
|
"sensibo",
|
||||||
"sentry",
|
"sentry",
|
||||||
"sharkiq",
|
"sharkiq",
|
||||||
|
|
11
mypy.ini
11
mypy.ini
|
@ -1364,6 +1364,17 @@ no_implicit_optional = true
|
||||||
warn_return_any = true
|
warn_return_any = true
|
||||||
warn_unreachable = true
|
warn_unreachable = true
|
||||||
|
|
||||||
|
[mypy-homeassistant.components.senseme.*]
|
||||||
|
check_untyped_defs = true
|
||||||
|
disallow_incomplete_defs = true
|
||||||
|
disallow_subclassing_any = true
|
||||||
|
disallow_untyped_calls = true
|
||||||
|
disallow_untyped_decorators = true
|
||||||
|
disallow_untyped_defs = true
|
||||||
|
no_implicit_optional = true
|
||||||
|
warn_return_any = true
|
||||||
|
warn_unreachable = true
|
||||||
|
|
||||||
[mypy-homeassistant.components.shelly.*]
|
[mypy-homeassistant.components.shelly.*]
|
||||||
check_untyped_defs = true
|
check_untyped_defs = true
|
||||||
disallow_incomplete_defs = true
|
disallow_incomplete_defs = true
|
||||||
|
|
|
@ -250,6 +250,9 @@ aiorecollect==1.0.8
|
||||||
# homeassistant.components.ridwell
|
# homeassistant.components.ridwell
|
||||||
aioridwell==2021.12.2
|
aioridwell==2021.12.2
|
||||||
|
|
||||||
|
# homeassistant.components.senseme
|
||||||
|
aiosenseme==0.5.5
|
||||||
|
|
||||||
# homeassistant.components.shelly
|
# homeassistant.components.shelly
|
||||||
aioshelly==1.0.7
|
aioshelly==1.0.7
|
||||||
|
|
||||||
|
|
|
@ -182,6 +182,9 @@ aiorecollect==1.0.8
|
||||||
# homeassistant.components.ridwell
|
# homeassistant.components.ridwell
|
||||||
aioridwell==2021.12.2
|
aioridwell==2021.12.2
|
||||||
|
|
||||||
|
# homeassistant.components.senseme
|
||||||
|
aiosenseme==0.5.5
|
||||||
|
|
||||||
# homeassistant.components.shelly
|
# homeassistant.components.shelly
|
||||||
aioshelly==1.0.7
|
aioshelly==1.0.7
|
||||||
|
|
||||||
|
|
117
tests/components/senseme/__init__.py
Normal file
117
tests/components/senseme/__init__.py
Normal file
|
@ -0,0 +1,117 @@
|
||||||
|
"""Tests for the SenseME integration."""
|
||||||
|
|
||||||
|
from contextlib import contextmanager
|
||||||
|
from unittest.mock import AsyncMock, MagicMock, patch
|
||||||
|
|
||||||
|
from aiosenseme import SensemeDevice, SensemeDiscovery
|
||||||
|
|
||||||
|
from homeassistant.components.senseme import config_flow
|
||||||
|
|
||||||
|
MOCK_NAME = "Haiku Fan"
|
||||||
|
MOCK_UUID = "77a6b7b3-925d-4695-a415-76d76dca4444"
|
||||||
|
MOCK_ADDRESS = "127.0.0.1"
|
||||||
|
|
||||||
|
device = MagicMock(auto_spec=SensemeDevice)
|
||||||
|
device.async_update = AsyncMock()
|
||||||
|
device.model = "Haiku Fan"
|
||||||
|
device.fan_speed_max = 7
|
||||||
|
device.mac = "aa:bb:cc:dd:ee:ff"
|
||||||
|
device.fan_dir = "REV"
|
||||||
|
device.room_name = "Main"
|
||||||
|
device.room_type = "Main"
|
||||||
|
device.fw_version = "1"
|
||||||
|
device.fan_autocomfort = "on"
|
||||||
|
device.fan_smartmode = "on"
|
||||||
|
device.fan_whoosh_mode = "on"
|
||||||
|
device.name = MOCK_NAME
|
||||||
|
device.uuid = MOCK_UUID
|
||||||
|
device.address = MOCK_ADDRESS
|
||||||
|
device.get_device_info = {
|
||||||
|
"name": MOCK_NAME,
|
||||||
|
"uuid": MOCK_UUID,
|
||||||
|
"mac": "20:F8:5E:92:5A:75",
|
||||||
|
"address": MOCK_ADDRESS,
|
||||||
|
"base_model": "FAN,HAIKU,HSERIES",
|
||||||
|
"has_light": False,
|
||||||
|
"has_sensor": True,
|
||||||
|
"is_fan": True,
|
||||||
|
"is_light": False,
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
device_alternate_ip = MagicMock(auto_spec=SensemeDevice)
|
||||||
|
device_alternate_ip.async_update = AsyncMock()
|
||||||
|
device_alternate_ip.model = "Haiku Fan"
|
||||||
|
device_alternate_ip.fan_speed_max = 7
|
||||||
|
device_alternate_ip.mac = "aa:bb:cc:dd:ee:ff"
|
||||||
|
device_alternate_ip.fan_dir = "REV"
|
||||||
|
device_alternate_ip.room_name = "Main"
|
||||||
|
device_alternate_ip.room_type = "Main"
|
||||||
|
device_alternate_ip.fw_version = "1"
|
||||||
|
device_alternate_ip.fan_autocomfort = "on"
|
||||||
|
device_alternate_ip.fan_smartmode = "on"
|
||||||
|
device_alternate_ip.fan_whoosh_mode = "on"
|
||||||
|
device_alternate_ip.name = MOCK_NAME
|
||||||
|
device_alternate_ip.uuid = MOCK_UUID
|
||||||
|
device_alternate_ip.address = "127.0.0.8"
|
||||||
|
device_alternate_ip.get_device_info = {
|
||||||
|
"name": MOCK_NAME,
|
||||||
|
"uuid": MOCK_UUID,
|
||||||
|
"mac": "20:F8:5E:92:5A:75",
|
||||||
|
"address": "127.0.0.8",
|
||||||
|
"base_model": "FAN,HAIKU,HSERIES",
|
||||||
|
"has_light": False,
|
||||||
|
"has_sensor": True,
|
||||||
|
"is_fan": True,
|
||||||
|
"is_light": False,
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
device2 = MagicMock(auto_spec=SensemeDevice)
|
||||||
|
device2.async_update = AsyncMock()
|
||||||
|
device2.model = "Haiku Fan"
|
||||||
|
device2.fan_speed_max = 7
|
||||||
|
device2.mac = "aa:bb:cc:dd:ee:ff"
|
||||||
|
device2.fan_dir = "FWD"
|
||||||
|
device2.room_name = "Main"
|
||||||
|
device2.room_type = "Main"
|
||||||
|
device2.fw_version = "1"
|
||||||
|
device2.fan_autocomfort = "on"
|
||||||
|
device2.fan_smartmode = "on"
|
||||||
|
device2.fan_whoosh_mode = "on"
|
||||||
|
device2.name = "Device 2"
|
||||||
|
device2.uuid = "uuid2"
|
||||||
|
device2.address = "127.0.0.2"
|
||||||
|
device2.get_device_info = {
|
||||||
|
"name": "Device 2",
|
||||||
|
"uuid": "uuid2",
|
||||||
|
"mac": "20:F8:5E:92:5A:76",
|
||||||
|
"address": "127.0.0.2",
|
||||||
|
"base_model": "FAN,HAIKU,HSERIES",
|
||||||
|
"has_light": True,
|
||||||
|
"has_sensor": True,
|
||||||
|
"is_fan": True,
|
||||||
|
"is_light": False,
|
||||||
|
}
|
||||||
|
|
||||||
|
MOCK_DEVICE = device
|
||||||
|
MOCK_DEVICE_ALTERNATE_IP = device_alternate_ip
|
||||||
|
MOCK_DEVICE2 = device2
|
||||||
|
|
||||||
|
|
||||||
|
def _patch_discovery(device=None, no_device=None):
|
||||||
|
"""Patch discovery."""
|
||||||
|
mock_senseme_discovery = MagicMock(auto_spec=SensemeDiscovery)
|
||||||
|
if not no_device:
|
||||||
|
mock_senseme_discovery.devices = [device or MOCK_DEVICE]
|
||||||
|
|
||||||
|
@contextmanager
|
||||||
|
def _patcher():
|
||||||
|
|
||||||
|
with patch.object(config_flow, "DISCOVER_TIMEOUT", 0), patch(
|
||||||
|
"homeassistant.components.senseme.discovery.SensemeDiscovery",
|
||||||
|
return_value=mock_senseme_discovery,
|
||||||
|
):
|
||||||
|
yield
|
||||||
|
|
||||||
|
return _patcher()
|
280
tests/components/senseme/test_config_flow.py
Normal file
280
tests/components/senseme/test_config_flow.py
Normal file
|
@ -0,0 +1,280 @@
|
||||||
|
"""Test the SenseME config flow."""
|
||||||
|
from unittest.mock import patch
|
||||||
|
|
||||||
|
from homeassistant import config_entries
|
||||||
|
from homeassistant.components.senseme.const import DOMAIN
|
||||||
|
from homeassistant.const import CONF_HOST, CONF_ID
|
||||||
|
from homeassistant.core import HomeAssistant
|
||||||
|
from homeassistant.data_entry_flow import (
|
||||||
|
RESULT_TYPE_ABORT,
|
||||||
|
RESULT_TYPE_CREATE_ENTRY,
|
||||||
|
RESULT_TYPE_FORM,
|
||||||
|
)
|
||||||
|
|
||||||
|
from . import (
|
||||||
|
MOCK_ADDRESS,
|
||||||
|
MOCK_DEVICE,
|
||||||
|
MOCK_DEVICE2,
|
||||||
|
MOCK_DEVICE_ALTERNATE_IP,
|
||||||
|
MOCK_UUID,
|
||||||
|
_patch_discovery,
|
||||||
|
)
|
||||||
|
|
||||||
|
from tests.common import MockConfigEntry
|
||||||
|
|
||||||
|
|
||||||
|
async def test_form_user(hass: HomeAssistant) -> None:
|
||||||
|
"""Test we get the form as a user."""
|
||||||
|
|
||||||
|
with _patch_discovery(), patch(
|
||||||
|
"homeassistant.components.senseme.async_setup_entry",
|
||||||
|
return_value=True,
|
||||||
|
) as mock_setup_entry:
|
||||||
|
result = await hass.config_entries.flow.async_init(
|
||||||
|
DOMAIN, context={"source": config_entries.SOURCE_USER}
|
||||||
|
)
|
||||||
|
assert result["type"] == RESULT_TYPE_FORM
|
||||||
|
assert not result["errors"]
|
||||||
|
|
||||||
|
result2 = await hass.config_entries.flow.async_configure(
|
||||||
|
result["flow_id"],
|
||||||
|
{
|
||||||
|
"device": MOCK_UUID,
|
||||||
|
},
|
||||||
|
)
|
||||||
|
await hass.async_block_till_done()
|
||||||
|
|
||||||
|
assert result2["type"] == RESULT_TYPE_CREATE_ENTRY
|
||||||
|
assert result2["title"] == "Haiku Fan"
|
||||||
|
assert result2["data"] == {
|
||||||
|
"info": MOCK_DEVICE.get_device_info,
|
||||||
|
}
|
||||||
|
assert len(mock_setup_entry.mock_calls) == 1
|
||||||
|
|
||||||
|
|
||||||
|
async def test_form_user_manual_entry(hass: HomeAssistant) -> None:
|
||||||
|
"""Test we get the form as a user with a discovery but user chooses manual."""
|
||||||
|
|
||||||
|
with _patch_discovery():
|
||||||
|
result = await hass.config_entries.flow.async_init(
|
||||||
|
DOMAIN, context={"source": config_entries.SOURCE_USER}
|
||||||
|
)
|
||||||
|
assert result["type"] == RESULT_TYPE_FORM
|
||||||
|
assert not result["errors"]
|
||||||
|
|
||||||
|
result2 = await hass.config_entries.flow.async_configure(
|
||||||
|
result["flow_id"],
|
||||||
|
{
|
||||||
|
"device": None,
|
||||||
|
},
|
||||||
|
)
|
||||||
|
await hass.async_block_till_done()
|
||||||
|
|
||||||
|
assert result2["type"] == RESULT_TYPE_FORM
|
||||||
|
assert result2["step_id"] == "manual"
|
||||||
|
|
||||||
|
with patch(
|
||||||
|
"homeassistant.components.senseme.config_flow.async_get_device_by_ip_address",
|
||||||
|
return_value=MOCK_DEVICE,
|
||||||
|
), patch(
|
||||||
|
"homeassistant.components.senseme.async_setup_entry",
|
||||||
|
return_value=True,
|
||||||
|
) as mock_setup_entry:
|
||||||
|
result3 = await hass.config_entries.flow.async_configure(
|
||||||
|
result["flow_id"],
|
||||||
|
{
|
||||||
|
CONF_HOST: MOCK_ADDRESS,
|
||||||
|
},
|
||||||
|
)
|
||||||
|
await hass.async_block_till_done()
|
||||||
|
|
||||||
|
assert result3["title"] == "Haiku Fan"
|
||||||
|
assert result3["data"] == {
|
||||||
|
"info": MOCK_DEVICE.get_device_info,
|
||||||
|
}
|
||||||
|
assert len(mock_setup_entry.mock_calls) == 1
|
||||||
|
|
||||||
|
|
||||||
|
async def test_form_user_no_discovery(hass: HomeAssistant) -> None:
|
||||||
|
"""Test we get the form as a user with no discovery."""
|
||||||
|
|
||||||
|
with _patch_discovery(no_device=True), patch(
|
||||||
|
"homeassistant.components.senseme.config_flow.async_get_device_by_ip_address",
|
||||||
|
return_value=MOCK_DEVICE,
|
||||||
|
), patch(
|
||||||
|
"homeassistant.components.senseme.async_setup_entry",
|
||||||
|
return_value=True,
|
||||||
|
) as mock_setup_entry:
|
||||||
|
result = await hass.config_entries.flow.async_init(
|
||||||
|
DOMAIN, context={"source": config_entries.SOURCE_USER}
|
||||||
|
)
|
||||||
|
assert result["type"] == RESULT_TYPE_FORM
|
||||||
|
assert not result["errors"]
|
||||||
|
|
||||||
|
result2 = await hass.config_entries.flow.async_configure(
|
||||||
|
result["flow_id"],
|
||||||
|
{
|
||||||
|
CONF_HOST: "not a valid address",
|
||||||
|
},
|
||||||
|
)
|
||||||
|
await hass.async_block_till_done()
|
||||||
|
|
||||||
|
assert result2["type"] == RESULT_TYPE_FORM
|
||||||
|
assert result2["step_id"] == "manual"
|
||||||
|
assert result2["errors"] == {CONF_HOST: "invalid_host"}
|
||||||
|
|
||||||
|
result3 = await hass.config_entries.flow.async_configure(
|
||||||
|
result2["flow_id"],
|
||||||
|
{
|
||||||
|
CONF_HOST: MOCK_ADDRESS,
|
||||||
|
},
|
||||||
|
)
|
||||||
|
await hass.async_block_till_done()
|
||||||
|
|
||||||
|
assert result3["title"] == "Haiku Fan"
|
||||||
|
assert result3["data"] == {
|
||||||
|
"info": MOCK_DEVICE.get_device_info,
|
||||||
|
}
|
||||||
|
assert len(mock_setup_entry.mock_calls) == 1
|
||||||
|
|
||||||
|
|
||||||
|
async def test_form_user_manual_entry_cannot_connect(hass: HomeAssistant) -> None:
|
||||||
|
"""Test we get the form as a user."""
|
||||||
|
|
||||||
|
with _patch_discovery():
|
||||||
|
result = await hass.config_entries.flow.async_init(
|
||||||
|
DOMAIN, context={"source": config_entries.SOURCE_USER}
|
||||||
|
)
|
||||||
|
assert result["type"] == RESULT_TYPE_FORM
|
||||||
|
assert not result["errors"]
|
||||||
|
|
||||||
|
result2 = await hass.config_entries.flow.async_configure(
|
||||||
|
result["flow_id"],
|
||||||
|
{
|
||||||
|
"device": None,
|
||||||
|
},
|
||||||
|
)
|
||||||
|
await hass.async_block_till_done()
|
||||||
|
|
||||||
|
assert result2["type"] == RESULT_TYPE_FORM
|
||||||
|
assert result2["step_id"] == "manual"
|
||||||
|
|
||||||
|
with patch(
|
||||||
|
"homeassistant.components.senseme.config_flow.async_get_device_by_ip_address",
|
||||||
|
return_value=None,
|
||||||
|
):
|
||||||
|
result3 = await hass.config_entries.flow.async_configure(
|
||||||
|
result["flow_id"],
|
||||||
|
{
|
||||||
|
CONF_HOST: MOCK_ADDRESS,
|
||||||
|
},
|
||||||
|
)
|
||||||
|
await hass.async_block_till_done()
|
||||||
|
|
||||||
|
assert result3["type"] == RESULT_TYPE_FORM
|
||||||
|
assert result3["step_id"] == "manual"
|
||||||
|
assert result3["errors"] == {CONF_HOST: "cannot_connect"}
|
||||||
|
|
||||||
|
|
||||||
|
async def test_discovery(hass: HomeAssistant) -> None:
|
||||||
|
"""Test we can setup a discovered device."""
|
||||||
|
entry = MockConfigEntry(
|
||||||
|
domain=DOMAIN,
|
||||||
|
data={
|
||||||
|
"info": MOCK_DEVICE2.get_device_info,
|
||||||
|
},
|
||||||
|
unique_id=MOCK_DEVICE2.uuid,
|
||||||
|
)
|
||||||
|
entry.add_to_hass(hass)
|
||||||
|
|
||||||
|
with _patch_discovery(), patch(
|
||||||
|
"homeassistant.components.senseme.async_get_device_by_device_info",
|
||||||
|
return_value=(True, MOCK_DEVICE2),
|
||||||
|
):
|
||||||
|
await hass.config_entries.async_setup(entry.entry_id)
|
||||||
|
await hass.async_block_till_done()
|
||||||
|
|
||||||
|
with _patch_discovery(), patch(
|
||||||
|
"homeassistant.components.senseme.async_setup_entry",
|
||||||
|
return_value=True,
|
||||||
|
) as mock_setup_entry:
|
||||||
|
result = await hass.config_entries.flow.async_init(
|
||||||
|
DOMAIN,
|
||||||
|
context={"source": config_entries.SOURCE_DISCOVERY},
|
||||||
|
data={CONF_ID: MOCK_UUID},
|
||||||
|
)
|
||||||
|
assert result["type"] == RESULT_TYPE_FORM
|
||||||
|
assert not result["errors"]
|
||||||
|
|
||||||
|
result2 = await hass.config_entries.flow.async_configure(
|
||||||
|
result["flow_id"],
|
||||||
|
{
|
||||||
|
"device": MOCK_UUID,
|
||||||
|
},
|
||||||
|
)
|
||||||
|
await hass.async_block_till_done()
|
||||||
|
|
||||||
|
assert result2["type"] == RESULT_TYPE_CREATE_ENTRY
|
||||||
|
assert result2["title"] == "Haiku Fan"
|
||||||
|
assert result2["data"] == {
|
||||||
|
"info": MOCK_DEVICE.get_device_info,
|
||||||
|
}
|
||||||
|
assert len(mock_setup_entry.mock_calls) == 1
|
||||||
|
|
||||||
|
|
||||||
|
async def test_discovery_existing_device_no_ip_change(hass: HomeAssistant) -> None:
|
||||||
|
"""Test we can setup a discovered device."""
|
||||||
|
entry = MockConfigEntry(
|
||||||
|
domain=DOMAIN,
|
||||||
|
data={
|
||||||
|
"info": MOCK_DEVICE.get_device_info,
|
||||||
|
},
|
||||||
|
unique_id=MOCK_DEVICE.uuid,
|
||||||
|
)
|
||||||
|
entry.add_to_hass(hass)
|
||||||
|
|
||||||
|
with _patch_discovery(), patch(
|
||||||
|
"homeassistant.components.senseme.async_get_device_by_device_info",
|
||||||
|
return_value=(True, MOCK_DEVICE),
|
||||||
|
):
|
||||||
|
await hass.config_entries.async_setup(entry.entry_id)
|
||||||
|
await hass.async_block_till_done()
|
||||||
|
|
||||||
|
with _patch_discovery():
|
||||||
|
result = await hass.config_entries.flow.async_init(
|
||||||
|
DOMAIN,
|
||||||
|
context={"source": config_entries.SOURCE_DISCOVERY},
|
||||||
|
data={CONF_ID: MOCK_UUID},
|
||||||
|
)
|
||||||
|
assert result["type"] == RESULT_TYPE_ABORT
|
||||||
|
assert result["reason"] == "already_configured"
|
||||||
|
|
||||||
|
|
||||||
|
async def test_discovery_existing_device_ip_change(hass: HomeAssistant) -> None:
|
||||||
|
"""Test a config entry ips get updated from discovery."""
|
||||||
|
entry = MockConfigEntry(
|
||||||
|
domain=DOMAIN,
|
||||||
|
data={
|
||||||
|
"info": MOCK_DEVICE.get_device_info,
|
||||||
|
},
|
||||||
|
unique_id=MOCK_DEVICE.uuid,
|
||||||
|
)
|
||||||
|
entry.add_to_hass(hass)
|
||||||
|
|
||||||
|
with _patch_discovery(device=MOCK_DEVICE_ALTERNATE_IP), patch(
|
||||||
|
"homeassistant.components.senseme.async_get_device_by_device_info",
|
||||||
|
return_value=(True, MOCK_DEVICE),
|
||||||
|
):
|
||||||
|
await hass.config_entries.async_setup(entry.entry_id)
|
||||||
|
await hass.async_block_till_done()
|
||||||
|
|
||||||
|
result = await hass.config_entries.flow.async_init(
|
||||||
|
DOMAIN,
|
||||||
|
context={"source": config_entries.SOURCE_DISCOVERY},
|
||||||
|
data={CONF_ID: MOCK_UUID},
|
||||||
|
)
|
||||||
|
await hass.async_block_till_done()
|
||||||
|
|
||||||
|
assert result["type"] == RESULT_TYPE_ABORT
|
||||||
|
assert result["reason"] == "already_configured"
|
||||||
|
assert entry.data["info"]["address"] == "127.0.0.8"
|
Loading…
Add table
Reference in a new issue