* Skeleton for Vogel's MotionMount support. * Generated updates. * Add validation of the discovered information. * Add manual configuration * Use a mac address as a unique id * Add tests for config_flow * Add a 'turn' sensor entity. * Add all needed sensors. * Add number and select entity for control of MotionMount * Update based on development checklist * Preset selector now updates when a preset is chosen * Fix adding presets selector to device * Remove irrelevant TODO * Bump python-MotionMount requirement * Invert direction of turn slider * Prepare for PR * Make sure entities have correct values when created * Use device's mac address as unique id for entities. * Fix missing files in .coveragerc * Remove typing ignore from device library. Improved typing also gave rise to the need to improve the callback mechanism * Improve typing * Convert property to shorthand form * Remove unneeded CONF_NAME in ConfigEntry * Add small comment * Refresh coordinator on notification from MotionMount * Use translation for entity * Bump python-MotionMount * Raise `ConfigEntryNotReady` when connect fails * Use local variable * Improve exception handling * Reduce duplicate code * Make better use of constants * Remove unneeded callback * Remove other occurrence of unneeded callback * Improve removal of suffix * Catch 'getaddrinfo' exception * Add config flow tests for invalid hostname * Abort if device with same hostname is already configured * Make sure we connect to a device with the same unique id as configured * Convert function names to snake_case * Remove unneeded commented-out code * Use tuple * Make us of config_entry id when mac is missing * Prevent update of entities when nothing changed * Don't store data in `hass.data` until we know we will proceed * Remove coordinator * Handle situation where mac is EMPTY_MAC * Disable polling * Fix failing hassfest * Avoid calling unique-id-less discovery handler for situations where we've an unique id
61 lines
2.1 KiB
Python
61 lines
2.1 KiB
Python
"""The Vogel's MotionMount integration."""
|
|
from __future__ import annotations
|
|
|
|
import socket
|
|
|
|
import motionmount
|
|
|
|
from homeassistant.config_entries import ConfigEntry
|
|
from homeassistant.const import CONF_HOST, CONF_PORT, Platform
|
|
from homeassistant.core import HomeAssistant
|
|
from homeassistant.exceptions import ConfigEntryNotReady
|
|
from homeassistant.helpers.device_registry import format_mac
|
|
|
|
from .const import DOMAIN, EMPTY_MAC
|
|
|
|
PLATFORMS: list[Platform] = [
|
|
Platform.NUMBER,
|
|
]
|
|
|
|
|
|
async def async_setup_entry(hass: HomeAssistant, entry: ConfigEntry) -> bool:
|
|
"""Set up Vogel's MotionMount from a config entry."""
|
|
|
|
host = entry.data[CONF_HOST]
|
|
|
|
# Create API instance
|
|
mm = motionmount.MotionMount(host, entry.data[CONF_PORT])
|
|
|
|
# Validate the API connection
|
|
try:
|
|
await mm.connect()
|
|
except (ConnectionError, TimeoutError, socket.gaierror) as ex:
|
|
raise ConfigEntryNotReady(f"Failed to connect to {host}") from ex
|
|
|
|
found_mac = format_mac(mm.mac.hex())
|
|
if found_mac not in (EMPTY_MAC, entry.unique_id):
|
|
# If the mac address of the device does not match the unique_id
|
|
# of the config entry, it likely means the DHCP lease has expired
|
|
# and the device has been assigned a new IP address. We need to
|
|
# wait for the next discovery to find the device at its new address
|
|
# and update the config entry so we do not mix up devices.
|
|
await mm.disconnect()
|
|
raise ConfigEntryNotReady(
|
|
f"Unexpected device found at {host}; expected {entry.unique_id}, found {found_mac}"
|
|
)
|
|
|
|
# Store an API object for your platforms to access
|
|
hass.data.setdefault(DOMAIN, {})[entry.entry_id] = mm
|
|
|
|
await hass.config_entries.async_forward_entry_setups(entry, PLATFORMS)
|
|
|
|
return True
|
|
|
|
|
|
async def async_unload_entry(hass: HomeAssistant, entry: ConfigEntry) -> bool:
|
|
"""Unload a config entry."""
|
|
if unload_ok := await hass.config_entries.async_unload_platforms(entry, PLATFORMS):
|
|
mm: motionmount.MotionMount = hass.data[DOMAIN].pop(entry.entry_id)
|
|
await mm.disconnect()
|
|
|
|
return unload_ok
|