185 lines
5.6 KiB
Python
185 lines
5.6 KiB
Python
"""Platform for binary sensor integration."""
|
|
from __future__ import annotations
|
|
|
|
from smarttub import SpaError, SpaReminder
|
|
import voluptuous as vol
|
|
|
|
from homeassistant.components.binary_sensor import (
|
|
DEVICE_CLASS_CONNECTIVITY,
|
|
DEVICE_CLASS_PROBLEM,
|
|
BinarySensorEntity,
|
|
)
|
|
from homeassistant.helpers import entity_platform
|
|
|
|
from .const import ATTR_ERRORS, ATTR_REMINDERS, DOMAIN, SMARTTUB_CONTROLLER
|
|
from .entity import SmartTubEntity, SmartTubSensorBase
|
|
|
|
# whether the reminder has been snoozed (bool)
|
|
ATTR_REMINDER_SNOOZED = "snoozed"
|
|
|
|
ATTR_ERROR_CODE = "error_code"
|
|
ATTR_ERROR_TITLE = "error_title"
|
|
ATTR_ERROR_DESCRIPTION = "error_description"
|
|
ATTR_ERROR_TYPE = "error_type"
|
|
ATTR_CREATED_AT = "created_at"
|
|
ATTR_UPDATED_AT = "updated_at"
|
|
|
|
# how many days to snooze the reminder for
|
|
ATTR_REMINDER_DAYS = "days"
|
|
RESET_REMINDER_SCHEMA = {
|
|
vol.Required(ATTR_REMINDER_DAYS): vol.All(
|
|
vol.Coerce(int), vol.Range(min=30, max=365)
|
|
)
|
|
}
|
|
SNOOZE_REMINDER_SCHEMA = {
|
|
vol.Required(ATTR_REMINDER_DAYS): vol.All(
|
|
vol.Coerce(int), vol.Range(min=10, max=120)
|
|
)
|
|
}
|
|
|
|
|
|
async def async_setup_entry(hass, entry, async_add_entities):
|
|
"""Set up binary sensor entities for the binary sensors in the tub."""
|
|
|
|
controller = hass.data[DOMAIN][entry.entry_id][SMARTTUB_CONTROLLER]
|
|
|
|
entities = []
|
|
for spa in controller.spas:
|
|
entities.append(SmartTubOnline(controller.coordinator, spa))
|
|
entities.append(SmartTubError(controller.coordinator, spa))
|
|
entities.extend(
|
|
SmartTubReminder(controller.coordinator, spa, reminder)
|
|
for reminder in controller.coordinator.data[spa.id][ATTR_REMINDERS].values()
|
|
)
|
|
|
|
async_add_entities(entities)
|
|
|
|
platform = entity_platform.current_platform.get()
|
|
|
|
platform.async_register_entity_service(
|
|
"snooze_reminder",
|
|
SNOOZE_REMINDER_SCHEMA,
|
|
"async_snooze",
|
|
)
|
|
platform.async_register_entity_service(
|
|
"reset_reminder",
|
|
RESET_REMINDER_SCHEMA,
|
|
"async_reset",
|
|
)
|
|
|
|
|
|
class SmartTubOnline(SmartTubSensorBase, BinarySensorEntity):
|
|
"""A binary sensor indicating whether the spa is currently online (connected to the cloud)."""
|
|
|
|
_attr_device_class = DEVICE_CLASS_CONNECTIVITY
|
|
|
|
def __init__(self, coordinator, spa):
|
|
"""Initialize the entity."""
|
|
super().__init__(coordinator, spa, "Online", "online")
|
|
|
|
@property
|
|
def entity_registry_enabled_default(self) -> bool:
|
|
"""Return if the entity should be enabled when first added to the entity registry.
|
|
|
|
This seems to be very noisy and not generally useful, so disable by default.
|
|
"""
|
|
return False
|
|
|
|
@property
|
|
def is_on(self) -> bool:
|
|
"""Return true if the binary sensor is on."""
|
|
return self._state is True
|
|
|
|
|
|
class SmartTubReminder(SmartTubEntity, BinarySensorEntity):
|
|
"""Reminders for maintenance actions."""
|
|
|
|
_attr_device_class = DEVICE_CLASS_PROBLEM
|
|
|
|
def __init__(self, coordinator, spa, reminder):
|
|
"""Initialize the entity."""
|
|
super().__init__(
|
|
coordinator,
|
|
spa,
|
|
f"{reminder.name.title()} Reminder",
|
|
)
|
|
self.reminder_id = reminder.id
|
|
|
|
@property
|
|
def unique_id(self):
|
|
"""Return a unique id for this sensor."""
|
|
return f"{self.spa.id}-reminder-{self.reminder_id}"
|
|
|
|
@property
|
|
def reminder(self) -> SpaReminder:
|
|
"""Return the underlying SpaReminder object for this entity."""
|
|
return self.coordinator.data[self.spa.id][ATTR_REMINDERS][self.reminder_id]
|
|
|
|
@property
|
|
def is_on(self) -> bool:
|
|
"""Return whether the specified maintenance action needs to be taken."""
|
|
return self.reminder.remaining_days == 0
|
|
|
|
@property
|
|
def extra_state_attributes(self):
|
|
"""Return the state attributes."""
|
|
return {
|
|
ATTR_REMINDER_SNOOZED: self.reminder.snoozed,
|
|
ATTR_REMINDER_DAYS: self.reminder.remaining_days,
|
|
}
|
|
|
|
async def async_snooze(self, days):
|
|
"""Snooze this reminder for the specified number of days."""
|
|
await self.reminder.snooze(days)
|
|
await self.coordinator.async_request_refresh()
|
|
|
|
async def async_reset(self, days):
|
|
"""Dismiss this reminder, and reset it to the specified number of days."""
|
|
await self.reminder.reset(days)
|
|
await self.coordinator.async_request_refresh()
|
|
|
|
|
|
class SmartTubError(SmartTubEntity, BinarySensorEntity):
|
|
"""Indicates whether an error code is present.
|
|
|
|
There may be 0 or more errors. If there are >0, we show the first one.
|
|
"""
|
|
|
|
_attr_device_class = DEVICE_CLASS_PROBLEM
|
|
|
|
def __init__(self, coordinator, spa):
|
|
"""Initialize the entity."""
|
|
super().__init__(
|
|
coordinator,
|
|
spa,
|
|
"Error",
|
|
)
|
|
|
|
@property
|
|
def error(self) -> SpaError | None:
|
|
"""Return the underlying SpaError object for this entity."""
|
|
errors = self.coordinator.data[self.spa.id][ATTR_ERRORS]
|
|
if len(errors) == 0:
|
|
return None
|
|
return errors[0]
|
|
|
|
@property
|
|
def is_on(self) -> bool:
|
|
"""Return true if an error is signaled."""
|
|
return self.error is not None
|
|
|
|
@property
|
|
def extra_state_attributes(self):
|
|
"""Return the state attributes."""
|
|
if (error := self.error) is None:
|
|
return {}
|
|
|
|
return {
|
|
ATTR_ERROR_CODE: error.code,
|
|
ATTR_ERROR_TITLE: error.title,
|
|
ATTR_ERROR_DESCRIPTION: error.description,
|
|
ATTR_ERROR_TYPE: error.error_type,
|
|
ATTR_CREATED_AT: error.created_at.isoformat(),
|
|
ATTR_UPDATED_AT: error.updated_at.isoformat(),
|
|
}
|