151 lines
4.8 KiB
Python
151 lines
4.8 KiB
Python
"""The myUplink integration."""
|
|
|
|
from __future__ import annotations
|
|
|
|
from http import HTTPStatus
|
|
import logging
|
|
|
|
from aiohttp import ClientError, ClientResponseError
|
|
import jwt
|
|
from myuplink import MyUplinkAPI, get_manufacturer, get_model, get_system_name
|
|
|
|
from homeassistant.const import Platform
|
|
from homeassistant.core import HomeAssistant, callback
|
|
from homeassistant.exceptions import ConfigEntryAuthFailed, ConfigEntryNotReady
|
|
from homeassistant.helpers import (
|
|
aiohttp_client,
|
|
config_entry_oauth2_flow,
|
|
device_registry as dr,
|
|
)
|
|
from homeassistant.helpers.device_registry import DeviceEntry
|
|
|
|
from .api import AsyncConfigEntryAuth
|
|
from .const import DOMAIN, OAUTH2_SCOPES
|
|
from .coordinator import MyUplinkConfigEntry, MyUplinkDataCoordinator
|
|
|
|
_LOGGER = logging.getLogger(__name__)
|
|
|
|
PLATFORMS: list[Platform] = [
|
|
Platform.BINARY_SENSOR,
|
|
Platform.NUMBER,
|
|
Platform.SELECT,
|
|
Platform.SENSOR,
|
|
Platform.SWITCH,
|
|
Platform.UPDATE,
|
|
]
|
|
|
|
|
|
async def async_setup_entry(
|
|
hass: HomeAssistant, config_entry: MyUplinkConfigEntry
|
|
) -> bool:
|
|
"""Set up myUplink from a config entry."""
|
|
|
|
implementation = (
|
|
await config_entry_oauth2_flow.async_get_config_entry_implementation(
|
|
hass, config_entry
|
|
)
|
|
)
|
|
session = config_entry_oauth2_flow.OAuth2Session(hass, config_entry, implementation)
|
|
auth = AsyncConfigEntryAuth(aiohttp_client.async_get_clientsession(hass), session)
|
|
|
|
try:
|
|
await auth.async_get_access_token()
|
|
except ClientResponseError as err:
|
|
if err.status in {HTTPStatus.UNAUTHORIZED, HTTPStatus.FORBIDDEN}:
|
|
raise ConfigEntryAuthFailed(
|
|
translation_domain=DOMAIN,
|
|
translation_key="config_entry_auth_failed",
|
|
) from err
|
|
raise ConfigEntryNotReady(
|
|
translation_domain=DOMAIN,
|
|
translation_key="config_entry_not_ready",
|
|
) from err
|
|
except ClientError as err:
|
|
raise ConfigEntryNotReady(
|
|
translation_domain=DOMAIN,
|
|
translation_key="config_entry_not_ready",
|
|
) from err
|
|
|
|
if set(config_entry.data["token"]["scope"].split(" ")) != set(OAUTH2_SCOPES):
|
|
raise ConfigEntryAuthFailed(
|
|
translation_domain=DOMAIN,
|
|
translation_key="incorrect_oauth2_scope",
|
|
)
|
|
|
|
# Setup MyUplinkAPI and coordinator for data fetch
|
|
api = MyUplinkAPI(auth)
|
|
coordinator = MyUplinkDataCoordinator(hass, config_entry, api)
|
|
await coordinator.async_config_entry_first_refresh()
|
|
config_entry.runtime_data = coordinator
|
|
|
|
# Update device registry
|
|
create_devices(hass, config_entry, coordinator)
|
|
|
|
await hass.config_entries.async_forward_entry_setups(config_entry, PLATFORMS)
|
|
|
|
return True
|
|
|
|
|
|
async def async_unload_entry(hass: HomeAssistant, entry: MyUplinkConfigEntry) -> bool:
|
|
"""Unload a config entry."""
|
|
return await hass.config_entries.async_unload_platforms(entry, PLATFORMS)
|
|
|
|
|
|
@callback
|
|
def create_devices(
|
|
hass: HomeAssistant,
|
|
config_entry: MyUplinkConfigEntry,
|
|
coordinator: MyUplinkDataCoordinator,
|
|
) -> None:
|
|
"""Update all devices."""
|
|
device_registry = dr.async_get(hass)
|
|
|
|
for system in coordinator.data.systems:
|
|
devices_in_system = [x.id for x in system.devices]
|
|
for device_id, device in coordinator.data.devices.items():
|
|
if device_id in devices_in_system:
|
|
device_registry.async_get_or_create(
|
|
config_entry_id=config_entry.entry_id,
|
|
identifiers={(DOMAIN, device_id)},
|
|
name=get_system_name(system),
|
|
manufacturer=get_manufacturer(device),
|
|
model=get_model(device),
|
|
sw_version=device.firmwareCurrent,
|
|
serial_number=device.product_serial_number,
|
|
)
|
|
|
|
|
|
async def async_remove_config_entry_device(
|
|
hass: HomeAssistant, config_entry: MyUplinkConfigEntry, device_entry: DeviceEntry
|
|
) -> bool:
|
|
"""Remove myuplink config entry from a device."""
|
|
|
|
myuplink_data = config_entry.runtime_data
|
|
return not device_entry.identifiers.intersection(
|
|
(DOMAIN, device_id) for device_id in myuplink_data.data.devices
|
|
)
|
|
|
|
|
|
async def async_migrate_entry(
|
|
hass: HomeAssistant, config_entry: MyUplinkConfigEntry
|
|
) -> bool:
|
|
"""Migrate old entry."""
|
|
|
|
# Use sub(ject) from access_token as unique_id
|
|
if config_entry.version == 1 and config_entry.minor_version == 1:
|
|
token = jwt.decode(
|
|
config_entry.data["token"]["access_token"],
|
|
options={"verify_signature": False},
|
|
)
|
|
uid = token["sub"]
|
|
hass.config_entries.async_update_entry(
|
|
config_entry, unique_id=uid, minor_version=2
|
|
)
|
|
_LOGGER.info(
|
|
"Migration to version %s.%s successful",
|
|
config_entry.version,
|
|
config_entry.minor_version,
|
|
)
|
|
|
|
return True
|