211 lines
6.7 KiB
Python
211 lines
6.7 KiB
Python
"""The La Marzocco integration."""
|
|
|
|
import asyncio
|
|
import logging
|
|
|
|
from packaging import version
|
|
from pylamarzocco import (
|
|
LaMarzoccoBluetoothClient,
|
|
LaMarzoccoCloudClient,
|
|
LaMarzoccoMachine,
|
|
)
|
|
from pylamarzocco.const import FirmwareType
|
|
from pylamarzocco.exceptions import AuthFail, RequestNotSuccessful
|
|
|
|
from homeassistant.components.bluetooth import async_discovered_service_info
|
|
from homeassistant.const import (
|
|
CONF_MAC,
|
|
CONF_PASSWORD,
|
|
CONF_TOKEN,
|
|
CONF_USERNAME,
|
|
Platform,
|
|
)
|
|
from homeassistant.core import HomeAssistant
|
|
from homeassistant.exceptions import ConfigEntryAuthFailed, ConfigEntryNotReady
|
|
from homeassistant.helpers import issue_registry as ir
|
|
from homeassistant.helpers.aiohttp_client import async_create_clientsession
|
|
|
|
from .const import CONF_USE_BLUETOOTH, DOMAIN
|
|
from .coordinator import (
|
|
LaMarzoccoConfigEntry,
|
|
LaMarzoccoConfigUpdateCoordinator,
|
|
LaMarzoccoRuntimeData,
|
|
LaMarzoccoScheduleUpdateCoordinator,
|
|
LaMarzoccoSettingsUpdateCoordinator,
|
|
LaMarzoccoStatisticsUpdateCoordinator,
|
|
)
|
|
|
|
PLATFORMS = [
|
|
Platform.BINARY_SENSOR,
|
|
Platform.BUTTON,
|
|
Platform.CALENDAR,
|
|
Platform.NUMBER,
|
|
Platform.SELECT,
|
|
Platform.SENSOR,
|
|
Platform.SWITCH,
|
|
Platform.UPDATE,
|
|
]
|
|
|
|
BT_MODEL_PREFIXES = ("MICRA", "MINI", "GS3")
|
|
|
|
_LOGGER = logging.getLogger(__name__)
|
|
|
|
|
|
async def async_setup_entry(hass: HomeAssistant, entry: LaMarzoccoConfigEntry) -> bool:
|
|
"""Set up La Marzocco as config entry."""
|
|
|
|
assert entry.unique_id
|
|
serial = entry.unique_id
|
|
|
|
cloud_client = LaMarzoccoCloudClient(
|
|
username=entry.data[CONF_USERNAME],
|
|
password=entry.data[CONF_PASSWORD],
|
|
client=async_create_clientsession(hass),
|
|
)
|
|
|
|
try:
|
|
settings = await cloud_client.get_thing_settings(serial)
|
|
except AuthFail as ex:
|
|
raise ConfigEntryAuthFailed(
|
|
translation_domain=DOMAIN, translation_key="authentication_failed"
|
|
) from ex
|
|
except (RequestNotSuccessful, TimeoutError) as ex:
|
|
_LOGGER.debug(ex, exc_info=True)
|
|
raise ConfigEntryNotReady(
|
|
translation_domain=DOMAIN, translation_key="api_error"
|
|
) from ex
|
|
|
|
gateway_version = version.parse(
|
|
settings.firmwares[FirmwareType.GATEWAY].build_version
|
|
)
|
|
|
|
if gateway_version < version.parse("v5.0.9"):
|
|
# incompatible gateway firmware, create an issue
|
|
ir.async_create_issue(
|
|
hass,
|
|
DOMAIN,
|
|
"unsupported_gateway_firmware",
|
|
is_fixable=False,
|
|
severity=ir.IssueSeverity.ERROR,
|
|
translation_key="unsupported_gateway_firmware",
|
|
translation_placeholders={"gateway_version": str(gateway_version)},
|
|
)
|
|
|
|
# initialize Bluetooth
|
|
bluetooth_client: LaMarzoccoBluetoothClient | None = None
|
|
if entry.options.get(CONF_USE_BLUETOOTH, True) and (
|
|
token := settings.ble_auth_token
|
|
):
|
|
if CONF_MAC not in entry.data:
|
|
for discovery_info in async_discovered_service_info(hass):
|
|
if (
|
|
(name := discovery_info.name)
|
|
and name.startswith(BT_MODEL_PREFIXES)
|
|
and name.split("_")[1] == serial
|
|
):
|
|
_LOGGER.debug("Found Bluetooth device, configuring with Bluetooth")
|
|
# found a device, add MAC address to config entry
|
|
hass.config_entries.async_update_entry(
|
|
entry,
|
|
data={
|
|
**entry.data,
|
|
CONF_MAC: discovery_info.address,
|
|
},
|
|
)
|
|
|
|
if not entry.data[CONF_TOKEN]:
|
|
# update the token in the config entry
|
|
hass.config_entries.async_update_entry(
|
|
entry,
|
|
data={
|
|
**entry.data,
|
|
CONF_TOKEN: token,
|
|
},
|
|
)
|
|
|
|
if CONF_MAC in entry.data:
|
|
_LOGGER.debug("Initializing Bluetooth device")
|
|
bluetooth_client = LaMarzoccoBluetoothClient(
|
|
address_or_ble_device=entry.data[CONF_MAC],
|
|
ble_token=token,
|
|
)
|
|
|
|
device = LaMarzoccoMachine(
|
|
serial_number=entry.unique_id,
|
|
cloud_client=cloud_client,
|
|
bluetooth_client=bluetooth_client,
|
|
)
|
|
|
|
coordinators = LaMarzoccoRuntimeData(
|
|
LaMarzoccoConfigUpdateCoordinator(hass, entry, device),
|
|
LaMarzoccoSettingsUpdateCoordinator(hass, entry, device),
|
|
LaMarzoccoScheduleUpdateCoordinator(hass, entry, device),
|
|
LaMarzoccoStatisticsUpdateCoordinator(hass, entry, device),
|
|
)
|
|
|
|
await asyncio.gather(
|
|
coordinators.config_coordinator.async_config_entry_first_refresh(),
|
|
coordinators.settings_coordinator.async_config_entry_first_refresh(),
|
|
coordinators.schedule_coordinator.async_config_entry_first_refresh(),
|
|
coordinators.statistics_coordinator.async_config_entry_first_refresh(),
|
|
)
|
|
|
|
entry.runtime_data = coordinators
|
|
|
|
await hass.config_entries.async_forward_entry_setups(entry, PLATFORMS)
|
|
|
|
return True
|
|
|
|
|
|
async def async_unload_entry(hass: HomeAssistant, entry: LaMarzoccoConfigEntry) -> bool:
|
|
"""Unload a config entry."""
|
|
return await hass.config_entries.async_unload_platforms(entry, PLATFORMS)
|
|
|
|
|
|
async def async_migrate_entry(
|
|
hass: HomeAssistant, entry: LaMarzoccoConfigEntry
|
|
) -> bool:
|
|
"""Migrate config entry."""
|
|
if entry.version > 3:
|
|
# guard against downgrade from a future version
|
|
return False
|
|
|
|
if entry.version == 1:
|
|
_LOGGER.error(
|
|
"Migration from version 1 is no longer supported, please remove and re-add the integration"
|
|
)
|
|
return False
|
|
|
|
if entry.version == 2:
|
|
cloud_client = LaMarzoccoCloudClient(
|
|
username=entry.data[CONF_USERNAME],
|
|
password=entry.data[CONF_PASSWORD],
|
|
)
|
|
try:
|
|
things = await cloud_client.list_things()
|
|
except (AuthFail, RequestNotSuccessful) as exc:
|
|
_LOGGER.error("Migration failed with error %s", exc)
|
|
return False
|
|
v3_data = {
|
|
CONF_USERNAME: entry.data[CONF_USERNAME],
|
|
CONF_PASSWORD: entry.data[CONF_PASSWORD],
|
|
CONF_TOKEN: next(
|
|
(
|
|
thing.ble_auth_token
|
|
for thing in things
|
|
if thing.serial_number == entry.unique_id
|
|
),
|
|
None,
|
|
),
|
|
}
|
|
if CONF_MAC in entry.data:
|
|
v3_data[CONF_MAC] = entry.data[CONF_MAC]
|
|
hass.config_entries.async_update_entry(
|
|
entry,
|
|
data=v3_data,
|
|
version=3,
|
|
)
|
|
_LOGGER.debug("Migrated La Marzocco config entry to version 2")
|
|
|
|
return True
|