core/homeassistant/components/smarttub/controller.py

133 lines
4.0 KiB
Python

"""Interface to the SmartTub API."""
import asyncio
from datetime import timedelta
import logging
from aiohttp import client_exceptions
import async_timeout
from smarttub import APIError, LoginFailed, SmartTub
from smarttub.api import Account
from homeassistant.const import CONF_EMAIL, CONF_PASSWORD
from homeassistant.core import callback
from homeassistant.exceptions import ConfigEntryAuthFailed, ConfigEntryNotReady
from homeassistant.helpers import device_registry as dr
from homeassistant.helpers.aiohttp_client import async_get_clientsession
from homeassistant.helpers.update_coordinator import DataUpdateCoordinator, UpdateFailed
from .const import (
ATTR_ERRORS,
ATTR_LIGHTS,
ATTR_PUMPS,
ATTR_REMINDERS,
ATTR_STATUS,
DOMAIN,
POLLING_TIMEOUT,
SCAN_INTERVAL,
)
from .helpers import get_spa_name
_LOGGER = logging.getLogger(__name__)
class SmartTubController:
"""Interface between Home Assistant and the SmartTub API."""
def __init__(self, hass):
"""Initialize an interface to SmartTub."""
self._hass = hass
self._account = None
self.spas = set()
self.coordinator = None
async def async_setup_entry(self, entry):
"""Perform initial setup.
Authenticate, query static state, set up polling, and otherwise make
ready for normal operations .
"""
try:
self._account = await self.login(
entry.data[CONF_EMAIL], entry.data[CONF_PASSWORD]
)
except LoginFailed as ex:
# credentials were changed or invalidated, we need new ones
raise ConfigEntryAuthFailed from ex
except (
asyncio.TimeoutError,
client_exceptions.ClientOSError,
client_exceptions.ServerDisconnectedError,
client_exceptions.ContentTypeError,
) as err:
raise ConfigEntryNotReady from err
self.spas = await self._account.get_spas()
self.coordinator = DataUpdateCoordinator(
self._hass,
_LOGGER,
name=DOMAIN,
update_method=self.async_update_data,
update_interval=timedelta(seconds=SCAN_INTERVAL),
)
await self.coordinator.async_refresh()
self.async_register_devices(entry)
return True
async def async_update_data(self):
"""Query the API and return the new state."""
data = {}
try:
async with async_timeout.timeout(POLLING_TIMEOUT):
for spa in self.spas:
data[spa.id] = await self._get_spa_data(spa)
except APIError as err:
raise UpdateFailed(err) from err
return data
async def _get_spa_data(self, spa):
full_status, reminders, errors = await asyncio.gather(
spa.get_status_full(),
spa.get_reminders(),
spa.get_errors(),
)
return {
ATTR_STATUS: full_status,
ATTR_PUMPS: {pump.id: pump for pump in full_status.pumps},
ATTR_LIGHTS: {light.zone: light for light in full_status.lights},
ATTR_REMINDERS: {reminder.id: reminder for reminder in reminders},
ATTR_ERRORS: errors,
}
@callback
def async_register_devices(self, entry):
"""Register devices with the device registry for all spas."""
device_registry = dr.async_get(self._hass)
for spa in self.spas:
device_registry.async_get_or_create(
config_entry_id=entry.entry_id,
identifiers={(DOMAIN, spa.id)},
manufacturer=spa.brand,
name=get_spa_name(spa),
model=spa.model,
)
async def login(self, email, password) -> Account:
"""Retrieve the account corresponding to the specified email and password.
Returns None if the credentials are invalid.
"""
api = SmartTub(async_get_clientsession(self._hass))
await api.login(email, password)
return await api.get_account()