core/homeassistant/components/screenlogic/__init__.py

191 lines
6.2 KiB
Python

"""The Screenlogic integration."""
import logging
from typing import Any
from screenlogicpy import ScreenLogicError, ScreenLogicGateway
from screenlogicpy.const.common import ScreenLogicConnectionError
from screenlogicpy.const.data import SHARED_VALUES
from homeassistant.config_entries import ConfigEntry
from homeassistant.const import Platform
from homeassistant.core import HomeAssistant
from homeassistant.exceptions import ConfigEntryNotReady
from homeassistant.helpers import config_validation as cv, entity_registry as er
from homeassistant.helpers.typing import ConfigType
from homeassistant.util import slugify
from .const import DOMAIN
from .coordinator import ScreenlogicDataUpdateCoordinator, async_get_connect_info
from .data import ENTITY_MIGRATIONS
from .services import async_load_screenlogic_services
from .util import generate_unique_id
type ScreenLogicConfigEntry = ConfigEntry[ScreenlogicDataUpdateCoordinator]
_LOGGER = logging.getLogger(__name__)
REQUEST_REFRESH_DELAY = 2
HEATER_COOLDOWN_DELAY = 6
# These seem to be constant across all controller models
PRIMARY_CIRCUIT_IDS = [500, 505] # [Spa, Pool]
PLATFORMS = [
Platform.BINARY_SENSOR,
Platform.CLIMATE,
Platform.LIGHT,
Platform.NUMBER,
Platform.SENSOR,
Platform.SWITCH,
]
CONFIG_SCHEMA = cv.config_entry_only_config_schema(DOMAIN)
async def async_setup(hass: HomeAssistant, config: ConfigType) -> bool:
"""Set up Screenlogic."""
async_load_screenlogic_services(hass)
return True
async def async_setup_entry(hass: HomeAssistant, entry: ScreenLogicConfigEntry) -> bool:
"""Set up Screenlogic from a config entry."""
await _async_migrate_entries(hass, entry)
gateway = ScreenLogicGateway()
connect_info = await async_get_connect_info(hass, entry)
try:
await gateway.async_connect(**connect_info)
await gateway.async_update()
except (ScreenLogicConnectionError, ScreenLogicError) as ex:
raise ConfigEntryNotReady(ex.msg) from ex
coordinator = ScreenlogicDataUpdateCoordinator(
hass, config_entry=entry, gateway=gateway
)
await coordinator.async_config_entry_first_refresh()
entry.async_on_unload(entry.add_update_listener(async_update_listener))
entry.runtime_data = coordinator
await hass.config_entries.async_forward_entry_setups(entry, PLATFORMS)
return True
async def async_unload_entry(
hass: HomeAssistant, entry: ScreenLogicConfigEntry
) -> bool:
"""Unload a config entry."""
unload_ok = await hass.config_entries.async_unload_platforms(entry, PLATFORMS)
if unload_ok:
coordinator = entry.runtime_data
await coordinator.gateway.async_disconnect()
return unload_ok
async def async_update_listener(
hass: HomeAssistant, entry: ScreenLogicConfigEntry
) -> None:
"""Handle options update."""
await hass.config_entries.async_reload(entry.entry_id)
async def _async_migrate_entries(
hass: HomeAssistant, config_entry: ScreenLogicConfigEntry
) -> None:
"""Migrate to new entity names."""
entity_registry = er.async_get(hass)
for entry in er.async_entries_for_config_entry(
entity_registry, config_entry.entry_id
):
source_mac, source_key = entry.unique_id.split("_", 1)
source_index = None
if (
len(key_parts := source_key.rsplit("_", 1)) == 2
and key_parts[1].isdecimal()
):
source_key, source_index = key_parts
_LOGGER.debug(
"Checking migration status for '%s' against key '%s'",
entry.unique_id,
source_key,
)
if source_key not in ENTITY_MIGRATIONS:
continue
_LOGGER.debug(
"Evaluating migration of '%s' from migration key '%s'",
entry.entity_id,
source_key,
)
migrations = ENTITY_MIGRATIONS[source_key]
updates: dict[str, Any] = {}
new_key = migrations["new_key"]
if new_key in SHARED_VALUES:
if (device := migrations.get("device")) is None:
_LOGGER.debug(
"Shared key '%s' is missing required migration data 'device'",
new_key,
)
continue
if device == "pump" and source_index is None:
_LOGGER.debug(
"Unable to parse 'source_index' from existing unique_id for pump entity '%s'",
source_key,
)
continue
new_unique_id = (
f"{source_mac}_{generate_unique_id(device, source_index, new_key)}"
)
else:
new_unique_id = entry.unique_id.replace(source_key, new_key)
if new_unique_id and new_unique_id != entry.unique_id:
if existing_entity_id := entity_registry.async_get_entity_id(
entry.domain, entry.platform, new_unique_id
):
_LOGGER.debug(
"Cannot migrate '%s' to unique_id '%s', already exists for entity '%s'. Aborting",
entry.unique_id,
new_unique_id,
existing_entity_id,
)
continue
updates["new_unique_id"] = new_unique_id
if (old_name := migrations.get("old_name")) is not None:
new_name = migrations["new_name"]
if (s_old_name := slugify(old_name)) in entry.entity_id:
new_entity_id = entry.entity_id.replace(s_old_name, slugify(new_name))
if new_entity_id and new_entity_id != entry.entity_id:
updates["new_entity_id"] = new_entity_id
if entry.original_name and old_name in entry.original_name:
new_original_name = entry.original_name.replace(old_name, new_name)
if new_original_name and new_original_name != entry.original_name:
updates["original_name"] = new_original_name
if updates:
_LOGGER.debug(
"Migrating entity '%s' unique_id from '%s' to '%s'",
entry.entity_id,
entry.unique_id,
new_unique_id,
)
entity_registry.async_update_entity(entry.entity_id, **updates)