2020-01-27 17:12:18 +00:00
|
|
|
"""Platform for Garmin Connect integration."""
|
2021-03-18 07:02:55 +00:00
|
|
|
from __future__ import annotations
|
|
|
|
|
2020-01-27 17:12:18 +00:00
|
|
|
import logging
|
|
|
|
|
2021-06-29 14:39:25 +00:00
|
|
|
from garminconnect_ha import (
|
2020-01-27 17:12:18 +00:00
|
|
|
GarminConnectAuthenticationError,
|
|
|
|
GarminConnectConnectionError,
|
|
|
|
GarminConnectTooManyRequestsError,
|
|
|
|
)
|
|
|
|
|
2021-03-22 18:45:17 +00:00
|
|
|
from homeassistant.components.sensor import SensorEntity
|
2020-01-27 17:12:18 +00:00
|
|
|
from homeassistant.config_entries import ConfigEntry
|
|
|
|
from homeassistant.const import ATTR_ATTRIBUTION, CONF_ID
|
2021-04-23 08:34:02 +00:00
|
|
|
from homeassistant.core import HomeAssistant
|
2021-05-01 22:37:19 +00:00
|
|
|
from homeassistant.helpers.entity import DeviceInfo
|
2020-01-27 17:12:18 +00:00
|
|
|
|
2020-10-23 21:01:29 +00:00
|
|
|
from .alarm_util import calculate_next_active_alarms
|
2020-01-27 17:12:18 +00:00
|
|
|
from .const import ATTRIBUTION, DOMAIN, GARMIN_ENTITY_LIST
|
|
|
|
|
|
|
|
_LOGGER = logging.getLogger(__name__)
|
|
|
|
|
|
|
|
|
|
|
|
async def async_setup_entry(
|
2021-04-23 08:34:02 +00:00
|
|
|
hass: HomeAssistant, entry: ConfigEntry, async_add_entities
|
2020-01-27 17:12:18 +00:00
|
|
|
) -> None:
|
|
|
|
"""Set up Garmin Connect sensor based on a config entry."""
|
|
|
|
garmin_data = hass.data[DOMAIN][entry.entry_id]
|
|
|
|
unique_id = entry.data[CONF_ID]
|
|
|
|
|
|
|
|
try:
|
|
|
|
await garmin_data.async_update()
|
|
|
|
except (
|
|
|
|
GarminConnectConnectionError,
|
|
|
|
GarminConnectAuthenticationError,
|
|
|
|
GarminConnectTooManyRequestsError,
|
|
|
|
) as err:
|
2020-01-31 16:33:00 +00:00
|
|
|
_LOGGER.error("Error occurred during Garmin Connect Client update: %s", err)
|
2020-01-27 17:12:18 +00:00
|
|
|
except Exception: # pylint: disable=broad-except
|
2020-07-05 21:04:19 +00:00
|
|
|
_LOGGER.exception("Unknown error occurred during Garmin Connect Client update")
|
2020-01-27 17:12:18 +00:00
|
|
|
|
|
|
|
entities = []
|
|
|
|
for (
|
|
|
|
sensor_type,
|
|
|
|
(name, unit, icon, device_class, enabled_by_default),
|
|
|
|
) in GARMIN_ENTITY_LIST.items():
|
|
|
|
|
|
|
|
_LOGGER.debug(
|
|
|
|
"Registering entity: %s, %s, %s, %s, %s, %s",
|
|
|
|
sensor_type,
|
|
|
|
name,
|
|
|
|
unit,
|
|
|
|
icon,
|
|
|
|
device_class,
|
|
|
|
enabled_by_default,
|
|
|
|
)
|
|
|
|
entities.append(
|
|
|
|
GarminConnectSensor(
|
|
|
|
garmin_data,
|
|
|
|
unique_id,
|
|
|
|
sensor_type,
|
|
|
|
name,
|
|
|
|
unit,
|
|
|
|
icon,
|
|
|
|
device_class,
|
|
|
|
enabled_by_default,
|
|
|
|
)
|
|
|
|
)
|
|
|
|
|
|
|
|
async_add_entities(entities, True)
|
|
|
|
|
|
|
|
|
2021-03-22 18:45:17 +00:00
|
|
|
class GarminConnectSensor(SensorEntity):
|
2020-01-27 17:12:18 +00:00
|
|
|
"""Representation of a Garmin Connect Sensor."""
|
|
|
|
|
|
|
|
def __init__(
|
|
|
|
self,
|
|
|
|
data,
|
|
|
|
unique_id,
|
|
|
|
sensor_type,
|
|
|
|
name,
|
|
|
|
unit,
|
|
|
|
icon,
|
|
|
|
device_class,
|
|
|
|
enabled_default: bool = True,
|
|
|
|
):
|
|
|
|
"""Initialize."""
|
|
|
|
self._data = data
|
|
|
|
self._unique_id = unique_id
|
|
|
|
self._type = sensor_type
|
|
|
|
self._name = name
|
|
|
|
self._unit = unit
|
|
|
|
self._icon = icon
|
|
|
|
self._device_class = device_class
|
|
|
|
self._enabled_default = enabled_default
|
|
|
|
self._available = True
|
|
|
|
self._state = None
|
|
|
|
|
|
|
|
@property
|
|
|
|
def name(self):
|
|
|
|
"""Return the name of the sensor."""
|
|
|
|
return self._name
|
|
|
|
|
|
|
|
@property
|
|
|
|
def icon(self):
|
|
|
|
"""Return the icon to use in the frontend."""
|
|
|
|
return self._icon
|
|
|
|
|
|
|
|
@property
|
|
|
|
def state(self):
|
|
|
|
"""Return the state of the sensor."""
|
|
|
|
return self._state
|
|
|
|
|
|
|
|
@property
|
|
|
|
def unique_id(self) -> str:
|
|
|
|
"""Return the unique ID for this sensor."""
|
|
|
|
return f"{self._unique_id}_{self._type}"
|
|
|
|
|
|
|
|
@property
|
|
|
|
def unit_of_measurement(self):
|
|
|
|
"""Return the unit of measurement."""
|
|
|
|
return self._unit
|
|
|
|
|
|
|
|
@property
|
2021-03-11 15:57:47 +00:00
|
|
|
def extra_state_attributes(self):
|
2020-01-27 17:12:18 +00:00
|
|
|
"""Return attributes for sensor."""
|
2020-10-06 16:08:53 +00:00
|
|
|
if not self._data.data:
|
|
|
|
return {}
|
2020-10-23 21:01:29 +00:00
|
|
|
attributes = {
|
2020-10-06 16:08:53 +00:00
|
|
|
"source": self._data.data["source"],
|
|
|
|
"last_synced": self._data.data["lastSyncTimestampGMT"],
|
|
|
|
ATTR_ATTRIBUTION: ATTRIBUTION,
|
|
|
|
}
|
2020-10-23 21:01:29 +00:00
|
|
|
if self._type == "nextAlarm":
|
|
|
|
attributes["next_alarms"] = calculate_next_active_alarms(
|
|
|
|
self._data.data[self._type]
|
|
|
|
)
|
|
|
|
return attributes
|
2020-01-27 17:12:18 +00:00
|
|
|
|
|
|
|
@property
|
2021-05-01 22:37:19 +00:00
|
|
|
def device_info(self) -> DeviceInfo:
|
2020-01-27 17:12:18 +00:00
|
|
|
"""Return device information."""
|
|
|
|
return {
|
|
|
|
"identifiers": {(DOMAIN, self._unique_id)},
|
|
|
|
"name": "Garmin Connect",
|
|
|
|
"manufacturer": "Garmin Connect",
|
|
|
|
}
|
|
|
|
|
|
|
|
@property
|
|
|
|
def entity_registry_enabled_default(self) -> bool:
|
|
|
|
"""Return if the entity should be enabled when first added to the entity registry."""
|
|
|
|
return self._enabled_default
|
|
|
|
|
|
|
|
@property
|
|
|
|
def available(self) -> bool:
|
|
|
|
"""Return True if entity is available."""
|
|
|
|
return self._available
|
|
|
|
|
|
|
|
@property
|
|
|
|
def device_class(self):
|
|
|
|
"""Return the device class of the sensor."""
|
|
|
|
return self._device_class
|
|
|
|
|
|
|
|
async def async_update(self):
|
|
|
|
"""Update the data from Garmin Connect."""
|
|
|
|
if not self.enabled:
|
|
|
|
return
|
|
|
|
|
|
|
|
await self._data.async_update()
|
2020-02-15 15:53:10 +00:00
|
|
|
data = self._data.data
|
|
|
|
if not data:
|
2020-01-27 17:12:18 +00:00
|
|
|
_LOGGER.error("Didn't receive data from Garmin Connect")
|
|
|
|
return
|
2020-02-15 15:53:10 +00:00
|
|
|
if data.get(self._type) is None:
|
|
|
|
_LOGGER.debug("Entity type %s not set in fetched data", self._type)
|
|
|
|
self._available = False
|
2020-02-08 17:47:54 +00:00
|
|
|
return
|
2020-02-15 15:53:10 +00:00
|
|
|
self._available = True
|
|
|
|
|
|
|
|
if "Duration" in self._type or "Seconds" in self._type:
|
|
|
|
self._state = data[self._type] // 60
|
2020-03-24 17:29:40 +00:00
|
|
|
elif "Mass" in self._type or self._type == "weight":
|
|
|
|
self._state = round((data[self._type] / 1000), 2)
|
|
|
|
elif (
|
|
|
|
self._type == "bodyFat" or self._type == "bodyWater" or self._type == "bmi"
|
|
|
|
):
|
|
|
|
self._state = round(data[self._type], 2)
|
2020-10-23 21:01:29 +00:00
|
|
|
elif self._type == "nextAlarm":
|
|
|
|
active_alarms = calculate_next_active_alarms(data[self._type])
|
|
|
|
if active_alarms:
|
|
|
|
self._state = active_alarms[0]
|
|
|
|
else:
|
|
|
|
self._available = False
|
2020-02-15 15:53:10 +00:00
|
|
|
else:
|
|
|
|
self._state = data[self._type]
|
2020-01-27 17:12:18 +00:00
|
|
|
|
|
|
|
_LOGGER.debug(
|
|
|
|
"Entity %s set to state %s %s", self._type, self._state, self._unit
|
|
|
|
)
|