"""Common code for Withings.""" from __future__ import annotations import asyncio from collections.abc import Callable from dataclasses import dataclass import datetime from datetime import timedelta from enum import Enum, IntEnum from http import HTTPStatus import logging import re from typing import Any from aiohttp.web import Response import requests from withings_api import AbstractWithingsApi from withings_api.common import ( AuthFailedException, GetSleepSummaryField, MeasureGroupAttribs, MeasureType, MeasureTypes, NotifyAppli, SleepGetSummaryResponse, UnauthorizedException, query_measure_groups, ) from homeassistant.components import webhook from homeassistant.components.application_credentials import AuthImplementation from homeassistant.components.binary_sensor import DOMAIN as BINARY_SENSOR_DOMAIN from homeassistant.components.http import HomeAssistantView from homeassistant.components.sensor import DOMAIN as SENSOR_DOMAIN from homeassistant.config_entries import SOURCE_REAUTH, ConfigEntry from homeassistant.const import ( CONF_WEBHOOK_ID, MASS_KILOGRAMS, PERCENTAGE, SPEED_METERS_PER_SECOND, TIME_SECONDS, ) from homeassistant.core import CALLBACK_TYPE, HomeAssistant, callback from homeassistant.exceptions import HomeAssistantError from homeassistant.helpers import config_entry_oauth2_flow, entity_registry as er from homeassistant.helpers.config_entry_oauth2_flow import ( AUTH_CALLBACK_PATH, AbstractOAuth2Implementation, OAuth2Session, ) from homeassistant.helpers.entity import Entity from homeassistant.helpers.network import get_url from homeassistant.helpers.update_coordinator import DataUpdateCoordinator from homeassistant.util import dt from . import const from .const import Measurement _LOGGER = logging.getLogger(const.LOG_NAMESPACE) NOT_AUTHENTICATED_ERROR = re.compile( f"^{HTTPStatus.UNAUTHORIZED},.*", re.IGNORECASE, ) DATA_UPDATED_SIGNAL = "withings_entity_state_updated" MeasurementData = dict[Measurement, Any] class NotAuthenticatedError(HomeAssistantError): """Raise when not authenticated with the service.""" class ServiceError(HomeAssistantError): """Raise when the service has an error.""" class UpdateType(Enum): """Data update type.""" POLL = "poll" WEBHOOK = "webhook" @dataclass class WithingsAttribute: """Immutable class for describing withings sensor data.""" measurement: Measurement measute_type: Enum friendly_name: str unit_of_measurement: str icon: str | None platform: str enabled_by_default: bool update_type: UpdateType @dataclass class WithingsData: """Represents value and meta-data from the withings service.""" attribute: WithingsAttribute value: Any @dataclass class WebhookConfig: """Config for a webhook.""" id: str url: str enabled: bool @dataclass class StateData: """State data held by data manager for retrieval by entities.""" unique_id: str state: Any WITHINGS_ATTRIBUTES = [ WithingsAttribute( Measurement.WEIGHT_KG, MeasureType.WEIGHT, "Weight", MASS_KILOGRAMS, "mdi:weight-kilogram", SENSOR_DOMAIN, True, UpdateType.POLL, ), WithingsAttribute( Measurement.FAT_MASS_KG, MeasureType.FAT_MASS_WEIGHT, "Fat Mass", MASS_KILOGRAMS, "mdi:weight-kilogram", SENSOR_DOMAIN, True, UpdateType.POLL, ), WithingsAttribute( Measurement.FAT_FREE_MASS_KG, MeasureType.FAT_FREE_MASS, "Fat Free Mass", MASS_KILOGRAMS, "mdi:weight-kilogram", SENSOR_DOMAIN, True, UpdateType.POLL, ), WithingsAttribute( Measurement.MUSCLE_MASS_KG, MeasureType.MUSCLE_MASS, "Muscle Mass", MASS_KILOGRAMS, "mdi:weight-kilogram", SENSOR_DOMAIN, True, UpdateType.POLL, ), WithingsAttribute( Measurement.BONE_MASS_KG, MeasureType.BONE_MASS, "Bone Mass", MASS_KILOGRAMS, "mdi:weight-kilogram", SENSOR_DOMAIN, True, UpdateType.POLL, ), WithingsAttribute( Measurement.HEIGHT_M, MeasureType.HEIGHT, "Height", const.UOM_LENGTH_M, "mdi:ruler", SENSOR_DOMAIN, False, UpdateType.POLL, ), WithingsAttribute( Measurement.TEMP_C, MeasureType.TEMPERATURE, "Temperature", const.UOM_TEMP_C, "mdi:thermometer", SENSOR_DOMAIN, True, UpdateType.POLL, ), WithingsAttribute( Measurement.BODY_TEMP_C, MeasureType.BODY_TEMPERATURE, "Body Temperature", const.UOM_TEMP_C, "mdi:thermometer", SENSOR_DOMAIN, True, UpdateType.POLL, ), WithingsAttribute( Measurement.SKIN_TEMP_C, MeasureType.SKIN_TEMPERATURE, "Skin Temperature", const.UOM_TEMP_C, "mdi:thermometer", SENSOR_DOMAIN, True, UpdateType.POLL, ), WithingsAttribute( Measurement.FAT_RATIO_PCT, MeasureType.FAT_RATIO, "Fat Ratio", PERCENTAGE, None, SENSOR_DOMAIN, True, UpdateType.POLL, ), WithingsAttribute( Measurement.DIASTOLIC_MMHG, MeasureType.DIASTOLIC_BLOOD_PRESSURE, "Diastolic Blood Pressure", const.UOM_MMHG, None, SENSOR_DOMAIN, True, UpdateType.POLL, ), WithingsAttribute( Measurement.SYSTOLIC_MMGH, MeasureType.SYSTOLIC_BLOOD_PRESSURE, "Systolic Blood Pressure", const.UOM_MMHG, None, SENSOR_DOMAIN, True, UpdateType.POLL, ), WithingsAttribute( Measurement.HEART_PULSE_BPM, MeasureType.HEART_RATE, "Heart Pulse", const.UOM_BEATS_PER_MINUTE, "mdi:heart-pulse", SENSOR_DOMAIN, True, UpdateType.POLL, ), WithingsAttribute( Measurement.SPO2_PCT, MeasureType.SP02, "SP02", PERCENTAGE, None, SENSOR_DOMAIN, True, UpdateType.POLL, ), WithingsAttribute( Measurement.HYDRATION, MeasureType.HYDRATION, "Hydration", MASS_KILOGRAMS, "mdi:water", SENSOR_DOMAIN, False, UpdateType.POLL, ), WithingsAttribute( Measurement.PWV, MeasureType.PULSE_WAVE_VELOCITY, "Pulse Wave Velocity", SPEED_METERS_PER_SECOND, None, SENSOR_DOMAIN, True, UpdateType.POLL, ), WithingsAttribute( Measurement.SLEEP_BREATHING_DISTURBANCES_INTENSITY, GetSleepSummaryField.BREATHING_DISTURBANCES_INTENSITY, "Breathing disturbances intensity", "", "", SENSOR_DOMAIN, False, UpdateType.POLL, ), WithingsAttribute( Measurement.SLEEP_DEEP_DURATION_SECONDS, GetSleepSummaryField.DEEP_SLEEP_DURATION, "Deep sleep", TIME_SECONDS, "mdi:sleep", SENSOR_DOMAIN, False, UpdateType.POLL, ), WithingsAttribute( Measurement.SLEEP_TOSLEEP_DURATION_SECONDS, GetSleepSummaryField.DURATION_TO_SLEEP, "Time to sleep", TIME_SECONDS, "mdi:sleep", SENSOR_DOMAIN, False, UpdateType.POLL, ), WithingsAttribute( Measurement.SLEEP_TOWAKEUP_DURATION_SECONDS, GetSleepSummaryField.DURATION_TO_WAKEUP, "Time to wakeup", TIME_SECONDS, "mdi:sleep-off", SENSOR_DOMAIN, False, UpdateType.POLL, ), WithingsAttribute( Measurement.SLEEP_HEART_RATE_AVERAGE, GetSleepSummaryField.HR_AVERAGE, "Average heart rate", const.UOM_BEATS_PER_MINUTE, "mdi:heart-pulse", SENSOR_DOMAIN, False, UpdateType.POLL, ), WithingsAttribute( Measurement.SLEEP_HEART_RATE_MAX, GetSleepSummaryField.HR_MAX, "Maximum heart rate", const.UOM_BEATS_PER_MINUTE, "mdi:heart-pulse", SENSOR_DOMAIN, False, UpdateType.POLL, ), WithingsAttribute( Measurement.SLEEP_HEART_RATE_MIN, GetSleepSummaryField.HR_MIN, "Minimum heart rate", const.UOM_BEATS_PER_MINUTE, "mdi:heart-pulse", SENSOR_DOMAIN, False, UpdateType.POLL, ), WithingsAttribute( Measurement.SLEEP_LIGHT_DURATION_SECONDS, GetSleepSummaryField.LIGHT_SLEEP_DURATION, "Light sleep", TIME_SECONDS, "mdi:sleep", SENSOR_DOMAIN, False, UpdateType.POLL, ), WithingsAttribute( Measurement.SLEEP_REM_DURATION_SECONDS, GetSleepSummaryField.REM_SLEEP_DURATION, "REM sleep", TIME_SECONDS, "mdi:sleep", SENSOR_DOMAIN, False, UpdateType.POLL, ), WithingsAttribute( Measurement.SLEEP_RESPIRATORY_RATE_AVERAGE, GetSleepSummaryField.RR_AVERAGE, "Average respiratory rate", const.UOM_BREATHS_PER_MINUTE, None, SENSOR_DOMAIN, False, UpdateType.POLL, ), WithingsAttribute( Measurement.SLEEP_RESPIRATORY_RATE_MAX, GetSleepSummaryField.RR_MAX, "Maximum respiratory rate", const.UOM_BREATHS_PER_MINUTE, None, SENSOR_DOMAIN, False, UpdateType.POLL, ), WithingsAttribute( Measurement.SLEEP_RESPIRATORY_RATE_MIN, GetSleepSummaryField.RR_MIN, "Minimum respiratory rate", const.UOM_BREATHS_PER_MINUTE, None, SENSOR_DOMAIN, False, UpdateType.POLL, ), WithingsAttribute( Measurement.SLEEP_SCORE, GetSleepSummaryField.SLEEP_SCORE, "Sleep score", const.SCORE_POINTS, "mdi:medal", SENSOR_DOMAIN, False, UpdateType.POLL, ), WithingsAttribute( Measurement.SLEEP_SNORING, GetSleepSummaryField.SNORING, "Snoring", "", None, SENSOR_DOMAIN, False, UpdateType.POLL, ), WithingsAttribute( Measurement.SLEEP_SNORING_EPISODE_COUNT, GetSleepSummaryField.SNORING_EPISODE_COUNT, "Snoring episode count", "", None, SENSOR_DOMAIN, False, UpdateType.POLL, ), WithingsAttribute( Measurement.SLEEP_WAKEUP_COUNT, GetSleepSummaryField.WAKEUP_COUNT, "Wakeup count", const.UOM_FREQUENCY, "mdi:sleep-off", SENSOR_DOMAIN, False, UpdateType.POLL, ), WithingsAttribute( Measurement.SLEEP_WAKEUP_DURATION_SECONDS, GetSleepSummaryField.WAKEUP_DURATION, "Wakeup time", TIME_SECONDS, "mdi:sleep-off", SENSOR_DOMAIN, False, UpdateType.POLL, ), # Webhook measurements. WithingsAttribute( Measurement.IN_BED, NotifyAppli.BED_IN, "In bed", "", "mdi:bed", BINARY_SENSOR_DOMAIN, True, UpdateType.WEBHOOK, ), ] WITHINGS_MEASUREMENTS_MAP: dict[Measurement, WithingsAttribute] = { attr.measurement: attr for attr in WITHINGS_ATTRIBUTES } WITHINGS_MEASURE_TYPE_MAP: dict[ NotifyAppli | GetSleepSummaryField | MeasureType, WithingsAttribute ] = {attr.measute_type: attr for attr in WITHINGS_ATTRIBUTES} class ConfigEntryWithingsApi(AbstractWithingsApi): """Withing API that uses HA resources.""" def __init__( self, hass: HomeAssistant, config_entry: ConfigEntry, implementation: AbstractOAuth2Implementation, ) -> None: """Initialize object.""" self._hass = hass self._config_entry = config_entry self._implementation = implementation self.session = OAuth2Session(hass, config_entry, implementation) def _request( self, path: str, params: dict[str, Any], method: str = "GET" ) -> dict[str, Any]: """Perform an async request.""" asyncio.run_coroutine_threadsafe( self.session.async_ensure_token_valid(), self._hass.loop ).result() access_token = self._config_entry.data["token"]["access_token"] response = requests.request( method, f"{self.URL}/{path}", params=params, headers={"Authorization": f"Bearer {access_token}"}, ) return response.json() def json_message_response(message: str, message_code: int) -> Response: """Produce common json output.""" return HomeAssistantView.json({"message": message, "code": message_code}) class WebhookAvailability(IntEnum): """Represents various statuses of webhook availability.""" SUCCESS = 0 CONNECT_ERROR = 1 HTTP_ERROR = 2 NOT_WEBHOOK = 3 class WebhookUpdateCoordinator: """Coordinates webhook data updates across listeners.""" def __init__(self, hass: HomeAssistant, user_id: int) -> None: """Initialize the object.""" self._hass = hass self._user_id = user_id self._listeners: list[CALLBACK_TYPE] = [] self.data: MeasurementData = {} def async_add_listener(self, listener: CALLBACK_TYPE) -> Callable[[], None]: """Add a listener.""" self._listeners.append(listener) @callback def remove_listener() -> None: self.async_remove_listener(listener) return remove_listener def async_remove_listener(self, listener: CALLBACK_TYPE) -> None: """Remove a listener.""" self._listeners.remove(listener) def update_data(self, measurement: Measurement, value: Any) -> None: """Update the data object and notify listeners the data has changed.""" self.data[measurement] = value self.notify_data_changed() def notify_data_changed(self) -> None: """Notify all listeners the data has changed.""" for listener in self._listeners: listener() class DataManager: """Manage withing data.""" def __init__( self, hass: HomeAssistant, profile: str, api: ConfigEntryWithingsApi, user_id: int, webhook_config: WebhookConfig, ) -> None: """Initialize the data manager.""" self._hass = hass self._api = api self._user_id = user_id self._profile = profile self._webhook_config = webhook_config self._notify_subscribe_delay = datetime.timedelta(seconds=5) self._notify_unsubscribe_delay = datetime.timedelta(seconds=1) self._is_available = True self._cancel_interval_update_interval: CALLBACK_TYPE | None = None self._cancel_configure_webhook_subscribe_interval: CALLBACK_TYPE | None = None self._api_notification_id = f"withings_{self._user_id}" self.subscription_update_coordinator = DataUpdateCoordinator( hass, _LOGGER, name="subscription_update_coordinator", update_interval=timedelta(minutes=120), update_method=self.async_subscribe_webhook, ) self.poll_data_update_coordinator = DataUpdateCoordinator[ dict[MeasureType, Any] ]( hass, _LOGGER, name="poll_data_update_coordinator", update_interval=timedelta(minutes=120) if self._webhook_config.enabled else timedelta(minutes=10), update_method=self.async_get_all_data, ) self.webhook_update_coordinator = WebhookUpdateCoordinator( self._hass, self._user_id ) self._cancel_subscription_update: Callable[[], None] | None = None self._subscribe_webhook_run_count = 0 @property def webhook_config(self) -> WebhookConfig: """Get the webhook config.""" return self._webhook_config @property def user_id(self) -> int: """Get the user_id of the authenticated user.""" return self._user_id @property def profile(self) -> str: """Get the profile.""" return self._profile def async_start_polling_webhook_subscriptions(self) -> None: """Start polling webhook subscriptions (if enabled) to reconcile their setup.""" self.async_stop_polling_webhook_subscriptions() def empty_listener() -> None: pass self._cancel_subscription_update = ( self.subscription_update_coordinator.async_add_listener(empty_listener) ) def async_stop_polling_webhook_subscriptions(self) -> None: """Stop polling webhook subscriptions.""" if self._cancel_subscription_update: self._cancel_subscription_update() self._cancel_subscription_update = None async def _do_retry(self, func, attempts=3) -> Any: """Retry a function call. Withings' API occasionally and incorrectly throws errors. Retrying the call tends to work. """ exception = None for attempt in range(1, attempts + 1): _LOGGER.debug("Attempt %s of %s", attempt, attempts) try: return await func() except Exception as exception1: # pylint: disable=broad-except _LOGGER.debug( "Failed attempt %s of %s (%s)", attempt, attempts, exception1 ) # Make each backoff pause a little bit longer await asyncio.sleep(0.5 * attempt) exception = exception1 continue if exception: raise exception async def async_subscribe_webhook(self) -> None: """Subscribe the webhook to withings data updates.""" return await self._do_retry(self._async_subscribe_webhook) async def _async_subscribe_webhook(self) -> None: _LOGGER.debug("Configuring withings webhook") # On first startup, perform a fresh re-subscribe. Withings stops pushing data # if the webhook fails enough times but they don't remove the old subscription # config. This ensures the subscription is setup correctly and they start # pushing again. if self._subscribe_webhook_run_count == 0: _LOGGER.debug("Refreshing withings webhook configs") await self.async_unsubscribe_webhook() self._subscribe_webhook_run_count += 1 # Get the current webhooks. response = await self._hass.async_add_executor_job(self._api.notify_list) subscribed_applis = frozenset( profile.appli for profile in response.profiles if profile.callbackurl == self._webhook_config.url ) # Determine what subscriptions need to be created. ignored_applis = frozenset({NotifyAppli.USER, NotifyAppli.UNKNOWN}) to_add_applis = frozenset( appli for appli in NotifyAppli if appli not in subscribed_applis and appli not in ignored_applis ) # Subscribe to each one. for appli in to_add_applis: _LOGGER.debug( "Subscribing %s for %s in %s seconds", self._webhook_config.url, appli, self._notify_subscribe_delay.total_seconds(), ) # Withings will HTTP HEAD the callback_url and needs some downtime # between each call or there is a higher chance of failure. await asyncio.sleep(self._notify_subscribe_delay.total_seconds()) await self._hass.async_add_executor_job( self._api.notify_subscribe, self._webhook_config.url, appli ) async def async_unsubscribe_webhook(self) -> None: """Unsubscribe webhook from withings data updates.""" return await self._do_retry(self._async_unsubscribe_webhook) async def _async_unsubscribe_webhook(self) -> None: # Get the current webhooks. response = await self._hass.async_add_executor_job(self._api.notify_list) # Revoke subscriptions. for profile in response.profiles: _LOGGER.debug( "Unsubscribing %s for %s in %s seconds", profile.callbackurl, profile.appli, self._notify_unsubscribe_delay.total_seconds(), ) # Quick calls to Withings can result in the service returning errors. Give them # some time to cool down. await asyncio.sleep(self._notify_subscribe_delay.total_seconds()) await self._hass.async_add_executor_job( self._api.notify_revoke, profile.callbackurl, profile.appli ) async def async_get_all_data(self) -> dict[MeasureType, Any] | None: """Update all withings data.""" try: return await self._do_retry(self._async_get_all_data) except Exception as exception: # User is not authenticated. if isinstance( exception, (UnauthorizedException, AuthFailedException) ) or NOT_AUTHENTICATED_ERROR.match(str(exception)): context = { const.PROFILE: self._profile, "userid": self._user_id, "source": SOURCE_REAUTH, } # Check if reauth flow already exists. flow = next( iter( flow for flow in self._hass.config_entries.flow.async_progress_by_handler( const.DOMAIN ) if flow.context == context ), None, ) if flow: return # Start a reauth flow. await self._hass.config_entries.flow.async_init( const.DOMAIN, context=context, ) return raise exception async def _async_get_all_data(self) -> dict[MeasureType, Any] | None: _LOGGER.info("Updating all withings data") return { **await self.async_get_measures(), **await self.async_get_sleep_summary(), } async def async_get_measures(self) -> dict[MeasureType, Any]: """Get the measures data.""" _LOGGER.debug("Updating withings measures") now = dt.utcnow() startdate = now - datetime.timedelta(days=7) response = await self._hass.async_add_executor_job( self._api.measure_get_meas, None, None, startdate, now, None, startdate ) # Sort from oldest to newest. groups = sorted( query_measure_groups( response, MeasureTypes.ANY, MeasureGroupAttribs.UNAMBIGUOUS ), key=lambda group: group.created.datetime, reverse=False, ) return { WITHINGS_MEASURE_TYPE_MAP[measure.type].measurement: round( float(measure.value * pow(10, measure.unit)), 2 ) for group in groups for measure in group.measures if measure.type in WITHINGS_MEASURE_TYPE_MAP } async def async_get_sleep_summary(self) -> dict[MeasureType, Any]: """Get the sleep summary data.""" _LOGGER.debug("Updating withing sleep summary") now = dt.utcnow() yesterday = now - datetime.timedelta(days=1) yesterday_noon = datetime.datetime( yesterday.year, yesterday.month, yesterday.day, 12, 0, 0, 0, datetime.timezone.utc, ) def get_sleep_summary() -> SleepGetSummaryResponse: return self._api.sleep_get_summary( lastupdate=yesterday_noon, data_fields=[ GetSleepSummaryField.BREATHING_DISTURBANCES_INTENSITY, GetSleepSummaryField.DEEP_SLEEP_DURATION, GetSleepSummaryField.DURATION_TO_SLEEP, GetSleepSummaryField.DURATION_TO_WAKEUP, GetSleepSummaryField.HR_AVERAGE, GetSleepSummaryField.HR_MAX, GetSleepSummaryField.HR_MIN, GetSleepSummaryField.LIGHT_SLEEP_DURATION, GetSleepSummaryField.REM_SLEEP_DURATION, GetSleepSummaryField.RR_AVERAGE, GetSleepSummaryField.RR_MAX, GetSleepSummaryField.RR_MIN, GetSleepSummaryField.SLEEP_SCORE, GetSleepSummaryField.SNORING, GetSleepSummaryField.SNORING_EPISODE_COUNT, GetSleepSummaryField.WAKEUP_COUNT, GetSleepSummaryField.WAKEUP_DURATION, ], ) response = await self._hass.async_add_executor_job(get_sleep_summary) # Set the default to empty lists. raw_values: dict[GetSleepSummaryField, list[int]] = { field: [] for field in GetSleepSummaryField } # Collect the raw data. for serie in response.series: data = serie.data for field in GetSleepSummaryField: raw_values[field].append(dict(data)[field.value]) values: dict[GetSleepSummaryField, float] = {} def average(data: list[int]) -> float: return sum(data) / len(data) def set_value(field: GetSleepSummaryField, func: Callable) -> None: non_nones = [ value for value in raw_values.get(field, []) if value is not None ] values[field] = func(non_nones) if non_nones else None set_value(GetSleepSummaryField.BREATHING_DISTURBANCES_INTENSITY, average) set_value(GetSleepSummaryField.DEEP_SLEEP_DURATION, sum) set_value(GetSleepSummaryField.DURATION_TO_SLEEP, average) set_value(GetSleepSummaryField.DURATION_TO_WAKEUP, average) set_value(GetSleepSummaryField.HR_AVERAGE, average) set_value(GetSleepSummaryField.HR_MAX, average) set_value(GetSleepSummaryField.HR_MIN, average) set_value(GetSleepSummaryField.LIGHT_SLEEP_DURATION, sum) set_value(GetSleepSummaryField.REM_SLEEP_DURATION, sum) set_value(GetSleepSummaryField.RR_AVERAGE, average) set_value(GetSleepSummaryField.RR_MAX, average) set_value(GetSleepSummaryField.RR_MIN, average) set_value(GetSleepSummaryField.SLEEP_SCORE, max) set_value(GetSleepSummaryField.SNORING, average) set_value(GetSleepSummaryField.SNORING_EPISODE_COUNT, sum) set_value(GetSleepSummaryField.WAKEUP_COUNT, sum) set_value(GetSleepSummaryField.WAKEUP_DURATION, average) return { WITHINGS_MEASURE_TYPE_MAP[field].measurement: round(value, 4) if value is not None else None for field, value in values.items() } async def async_webhook_data_updated(self, data_category: NotifyAppli) -> None: """Handle scenario when data is updated from a webook.""" _LOGGER.debug("Withings webhook triggered") if data_category in { NotifyAppli.WEIGHT, NotifyAppli.CIRCULATORY, NotifyAppli.SLEEP, }: await self.poll_data_update_coordinator.async_request_refresh() elif data_category in {NotifyAppli.BED_IN, NotifyAppli.BED_OUT}: self.webhook_update_coordinator.update_data( Measurement.IN_BED, data_category == NotifyAppli.BED_IN ) def get_attribute_unique_id(attribute: WithingsAttribute, user_id: int) -> str: """Get a entity unique id for a user's attribute.""" return f"withings_{user_id}_{attribute.measurement.value}" async def async_get_entity_id( hass: HomeAssistant, attribute: WithingsAttribute, user_id: int ) -> str | None: """Get an entity id for a user's attribute.""" entity_registry = er.async_get(hass) unique_id = get_attribute_unique_id(attribute, user_id) entity_id = entity_registry.async_get_entity_id( attribute.platform, const.DOMAIN, unique_id ) if entity_id is None: _LOGGER.error("Cannot find entity id for unique_id: %s", unique_id) return None return entity_id class BaseWithingsSensor(Entity): """Base class for withings sensors.""" def __init__(self, data_manager: DataManager, attribute: WithingsAttribute) -> None: """Initialize the Withings sensor.""" self._data_manager = data_manager self._attribute = attribute self._profile = self._data_manager.profile self._user_id = self._data_manager.user_id self._name = f"Withings {self._attribute.measurement.value} {self._profile}" self._unique_id = get_attribute_unique_id(self._attribute, self._user_id) self._state_data: Any | None = None @property def should_poll(self) -> bool: """Return False to indicate HA should not poll for changes.""" return False @property def name(self) -> str: """Return the name of the sensor.""" return self._name @property def available(self) -> bool: """Return True if entity is available.""" if self._attribute.update_type == UpdateType.POLL: return self._data_manager.poll_data_update_coordinator.last_update_success if self._attribute.update_type == UpdateType.WEBHOOK: return self._data_manager.webhook_config.enabled and ( self._attribute.measurement in self._data_manager.webhook_update_coordinator.data ) return True @property def unique_id(self) -> str: """Return a unique, Home Assistant friendly identifier for this entity.""" return self._unique_id @property def icon(self) -> str: """Icon to use in the frontend, if any.""" return self._attribute.icon @property def entity_registry_enabled_default(self) -> bool: """Return if the entity should be enabled when first added to the entity registry.""" return self._attribute.enabled_by_default @callback def _on_poll_data_updated(self) -> None: self._update_state_data( self._data_manager.poll_data_update_coordinator.data or {} ) @callback def _on_webhook_data_updated(self) -> None: self._update_state_data( self._data_manager.webhook_update_coordinator.data or {} ) def _update_state_data(self, data: MeasurementData) -> None: """Update the state data.""" self._state_data = data.get(self._attribute.measurement) self.async_write_ha_state() async def async_added_to_hass(self) -> None: """Register update dispatcher.""" if self._attribute.update_type == UpdateType.POLL: self.async_on_remove( self._data_manager.poll_data_update_coordinator.async_add_listener( self._on_poll_data_updated ) ) self._on_poll_data_updated() elif self._attribute.update_type == UpdateType.WEBHOOK: self.async_on_remove( self._data_manager.webhook_update_coordinator.async_add_listener( self._on_webhook_data_updated ) ) self._on_webhook_data_updated() async def async_get_data_manager( hass: HomeAssistant, config_entry: ConfigEntry ) -> DataManager: """Get the data manager for a config entry.""" hass.data.setdefault(const.DOMAIN, {}) hass.data[const.DOMAIN].setdefault(config_entry.entry_id, {}) config_entry_data = hass.data[const.DOMAIN][config_entry.entry_id] if const.DATA_MANAGER not in config_entry_data: profile = config_entry.data.get(const.PROFILE) _LOGGER.debug("Creating withings data manager for profile: %s", profile) config_entry_data[const.DATA_MANAGER] = DataManager( hass, profile, ConfigEntryWithingsApi( hass=hass, config_entry=config_entry, implementation=await config_entry_oauth2_flow.async_get_config_entry_implementation( hass, config_entry ), ), config_entry.data["token"]["userid"], WebhookConfig( id=config_entry.data[CONF_WEBHOOK_ID], url=webhook.async_generate_url( hass, config_entry.data[CONF_WEBHOOK_ID] ), enabled=config_entry.data[const.CONF_USE_WEBHOOK], ), ) return config_entry_data[const.DATA_MANAGER] def get_data_manager_by_webhook_id( hass: HomeAssistant, webhook_id: str ) -> DataManager | None: """Get a data manager by it's webhook id.""" return next( iter( [ data_manager for data_manager in get_all_data_managers(hass) if data_manager.webhook_config.id == webhook_id ] ), None, ) def get_all_data_managers(hass: HomeAssistant) -> tuple[DataManager, ...]: """Get all configured data managers.""" return tuple( config_entry_data[const.DATA_MANAGER] for config_entry_data in hass.data[const.DOMAIN].values() if const.DATA_MANAGER in config_entry_data ) def async_remove_data_manager(hass: HomeAssistant, config_entry: ConfigEntry) -> None: """Remove a data manager for a config entry.""" del hass.data[const.DOMAIN][config_entry.entry_id][const.DATA_MANAGER] async def async_create_entities( hass: HomeAssistant, entry: ConfigEntry, create_func: Callable[[DataManager, WithingsAttribute], Entity], platform: str, ) -> list[Entity]: """Create withings entities from config entry.""" data_manager = await async_get_data_manager(hass, entry) return [ create_func(data_manager, attribute) for attribute in get_platform_attributes(platform) ] def get_platform_attributes(platform: str) -> tuple[WithingsAttribute, ...]: """Get withings attributes used for a specific platform.""" return tuple( attribute for attribute in WITHINGS_ATTRIBUTES if attribute.platform == platform ) class WithingsLocalOAuth2Implementation(AuthImplementation): """Oauth2 implementation that only uses the external url.""" @property def redirect_uri(self) -> str: """Return the redirect uri.""" url = get_url(self.hass, allow_internal=False, prefer_cloud=True) return f"{url}{AUTH_CALLBACK_PATH}" async def _token_request(self, data: dict) -> dict: """Make a token request and adapt Withings API reply.""" new_token = await super()._token_request(data) # Withings API returns habitual token data under json key "body": # { # "status": [{integer} Withings API response status], # "body": { # "access_token": [{string} Your new access_token], # "expires_in": [{integer} Access token expiry delay in seconds], # "token_type": [{string] HTTP Authorization Header format: Bearer], # "scope": [{string} Scopes the user accepted], # "refresh_token": [{string} Your new refresh_token], # "userid": [{string} The Withings ID of the user] # } # } # so we copy that to token root. if body := new_token.pop("body", None): new_token.update(body) return new_token async def async_resolve_external_data(self, external_data: Any) -> dict: """Resolve the authorization code to tokens.""" return await self._token_request( { "action": "requesttoken", "grant_type": "authorization_code", "code": external_data["code"], "redirect_uri": external_data["state"]["redirect_uri"], } ) async def _async_refresh_token(self, token: dict) -> dict: """Refresh tokens.""" new_token = await self._token_request( { "action": "requesttoken", "grant_type": "refresh_token", "client_id": self.client_id, "refresh_token": token["refresh_token"], } ) return {**token, **new_token}