198 lines
6.9 KiB
Python
198 lines
6.9 KiB
Python
"""Manager to set up IO with Crownstone devices for a config entry."""
|
|
from __future__ import annotations
|
|
|
|
import logging
|
|
from typing import Any
|
|
|
|
from crownstone_cloud import CrownstoneCloud
|
|
from crownstone_cloud.exceptions import (
|
|
CrownstoneAuthenticationError,
|
|
CrownstoneUnknownError,
|
|
)
|
|
from crownstone_sse import CrownstoneSSEAsync
|
|
from crownstone_uart import CrownstoneUart, UartEventBus
|
|
from crownstone_uart.Exceptions import UartException
|
|
|
|
from homeassistant.components import persistent_notification
|
|
from homeassistant.config_entries import ConfigEntry
|
|
from homeassistant.const import CONF_EMAIL, CONF_PASSWORD, EVENT_HOMEASSISTANT_STOP
|
|
from homeassistant.core import Event, HomeAssistant, callback
|
|
from homeassistant.exceptions import ConfigEntryNotReady
|
|
from homeassistant.helpers import aiohttp_client
|
|
from homeassistant.helpers.dispatcher import async_dispatcher_send
|
|
|
|
from .const import (
|
|
CONF_USB_PATH,
|
|
CONF_USB_SPHERE,
|
|
DOMAIN,
|
|
PLATFORMS,
|
|
PROJECT_NAME,
|
|
SSE_LISTENERS,
|
|
UART_LISTENERS,
|
|
)
|
|
from .helpers import get_port
|
|
from .listeners import setup_sse_listeners, setup_uart_listeners
|
|
|
|
_LOGGER = logging.getLogger(__name__)
|
|
|
|
|
|
class CrownstoneEntryManager:
|
|
"""Manage a Crownstone config entry."""
|
|
|
|
uart: CrownstoneUart | None = None
|
|
cloud: CrownstoneCloud
|
|
sse: CrownstoneSSEAsync
|
|
|
|
def __init__(self, hass: HomeAssistant, config_entry: ConfigEntry) -> None:
|
|
"""Initialize the hub."""
|
|
self.hass = hass
|
|
self.config_entry = config_entry
|
|
self.listeners: dict[str, Any] = {}
|
|
self.usb_sphere_id: str | None = None
|
|
|
|
async def async_setup(self) -> bool:
|
|
"""Set up a Crownstone config entry.
|
|
|
|
Returns True if the setup was successful.
|
|
"""
|
|
email = self.config_entry.data[CONF_EMAIL]
|
|
password = self.config_entry.data[CONF_PASSWORD]
|
|
|
|
self.cloud = CrownstoneCloud(
|
|
email=email,
|
|
password=password,
|
|
clientsession=aiohttp_client.async_get_clientsession(self.hass),
|
|
)
|
|
# Login & sync all user data
|
|
try:
|
|
await self.cloud.async_initialize()
|
|
except CrownstoneAuthenticationError as auth_err:
|
|
_LOGGER.error(
|
|
"Auth error during login with type: %s and message: %s",
|
|
auth_err.type,
|
|
auth_err.message,
|
|
)
|
|
return False
|
|
except CrownstoneUnknownError as unknown_err:
|
|
_LOGGER.error("Unknown error during login")
|
|
raise ConfigEntryNotReady from unknown_err
|
|
|
|
# A new clientsession is created because the default one does not cleanup on unload
|
|
self.sse = CrownstoneSSEAsync(
|
|
email=email,
|
|
password=password,
|
|
access_token=self.cloud.access_token,
|
|
websession=aiohttp_client.async_create_clientsession(self.hass),
|
|
project_name=PROJECT_NAME,
|
|
)
|
|
# Listen for events in the background, without task tracking
|
|
self.config_entry.async_create_background_task(
|
|
self.hass, self.async_process_events(self.sse), "crownstone-sse"
|
|
)
|
|
setup_sse_listeners(self)
|
|
|
|
# Set up a Crownstone USB only if path exists
|
|
if self.config_entry.options[CONF_USB_PATH] is not None:
|
|
await self.async_setup_usb()
|
|
|
|
# Save the sphere where the USB is located
|
|
# Makes HA aware of the Crownstone environment HA is placed in, a user can have multiple
|
|
self.usb_sphere_id = self.config_entry.options[CONF_USB_SPHERE]
|
|
|
|
await self.hass.config_entries.async_forward_entry_setups(
|
|
self.config_entry, PLATFORMS
|
|
)
|
|
|
|
# HA specific listeners
|
|
self.config_entry.async_on_unload(
|
|
self.config_entry.add_update_listener(_async_update_listener)
|
|
)
|
|
self.config_entry.async_on_unload(
|
|
self.hass.bus.async_listen_once(EVENT_HOMEASSISTANT_STOP, self.on_shutdown)
|
|
)
|
|
|
|
return True
|
|
|
|
async def async_process_events(self, sse_client: CrownstoneSSEAsync) -> None:
|
|
"""Asynchronous iteration of Crownstone SSE events."""
|
|
async with sse_client as client:
|
|
async for event in client:
|
|
if event is not None:
|
|
async_dispatcher_send(self.hass, f"{DOMAIN}_{event.type}", event)
|
|
|
|
async def async_setup_usb(self) -> None:
|
|
"""Attempt setup of a Crownstone usb dongle."""
|
|
# Trace by-id symlink back to the serial port
|
|
serial_port = await self.hass.async_add_executor_job(
|
|
get_port, self.config_entry.options[CONF_USB_PATH]
|
|
)
|
|
if serial_port is None:
|
|
return
|
|
|
|
self.uart = CrownstoneUart()
|
|
# UartException is raised when serial controller fails to open
|
|
try:
|
|
await self.uart.initialize_usb(serial_port)
|
|
except UartException:
|
|
self.uart = None
|
|
# Set entry options for usb to null
|
|
updated_options = self.config_entry.options.copy()
|
|
updated_options[CONF_USB_PATH] = None
|
|
updated_options[CONF_USB_SPHERE] = None
|
|
# Ensure that the user can configure an USB again from options
|
|
self.hass.config_entries.async_update_entry(
|
|
self.config_entry, options=updated_options
|
|
)
|
|
# Show notification to ensure the user knows the cloud is now used
|
|
persistent_notification.async_create(
|
|
self.hass,
|
|
(
|
|
"Setup of Crownstone USB dongle was unsuccessful on port"
|
|
f" {serial_port}.\n Crownstone Cloud will be used"
|
|
" to switch Crownstones.\n Please check if your"
|
|
" port is correct and set up the USB again from integration"
|
|
" options."
|
|
),
|
|
"Crownstone",
|
|
"crownstone_usb_dongle_setup",
|
|
)
|
|
return
|
|
|
|
setup_uart_listeners(self)
|
|
|
|
async def async_unload(self) -> bool:
|
|
"""Unload the current config entry."""
|
|
# Authentication failed
|
|
if self.cloud.cloud_data is None:
|
|
return True
|
|
|
|
self.sse.close_client()
|
|
for sse_unsub in self.listeners[SSE_LISTENERS]:
|
|
sse_unsub()
|
|
|
|
if self.uart:
|
|
self.uart.stop()
|
|
for subscription_id in self.listeners[UART_LISTENERS]:
|
|
UartEventBus.unsubscribe(subscription_id)
|
|
|
|
unload_ok = await self.hass.config_entries.async_unload_platforms(
|
|
self.config_entry, PLATFORMS
|
|
)
|
|
|
|
if unload_ok:
|
|
self.hass.data[DOMAIN].pop(self.config_entry.entry_id)
|
|
|
|
return unload_ok
|
|
|
|
@callback
|
|
def on_shutdown(self, _: Event) -> None:
|
|
"""Close all IO connections."""
|
|
self.sse.close_client()
|
|
if self.uart:
|
|
self.uart.stop()
|
|
|
|
|
|
async def _async_update_listener(hass: HomeAssistant, entry: ConfigEntry) -> None:
|
|
"""Handle options update."""
|
|
await hass.config_entries.async_reload(entry.entry_id)
|