"""Support for August binary sensors.""" from datetime import datetime, timedelta import logging from yalexs.activity import ACTION_DOORBELL_CALL_MISSED, SOURCE_PUBNUB, ActivityType from yalexs.lock import LockDoorStatus from yalexs.util import update_lock_detail_from_activity from homeassistant.components.binary_sensor import ( DEVICE_CLASS_CONNECTIVITY, DEVICE_CLASS_DOOR, DEVICE_CLASS_MOTION, DEVICE_CLASS_OCCUPANCY, BinarySensorEntity, ) from homeassistant.core import callback from homeassistant.helpers.event import async_call_later from .const import ACTIVITY_UPDATE_INTERVAL, DATA_AUGUST, DOMAIN from .entity import AugustEntityMixin _LOGGER = logging.getLogger(__name__) TIME_TO_DECLARE_DETECTION = timedelta(seconds=ACTIVITY_UPDATE_INTERVAL.total_seconds()) TIME_TO_RECHECK_DETECTION = timedelta( seconds=ACTIVITY_UPDATE_INTERVAL.total_seconds() * 3 ) def _retrieve_online_state(data, detail): """Get the latest state of the sensor.""" # The doorbell will go into standby mode when there is no motion # for a short while. It will wake by itself when needed so we need # to consider is available or we will not report motion or dings return detail.is_online or detail.is_standby def _retrieve_motion_state(data, detail): latest = data.activity_stream.get_latest_device_activity( detail.device_id, {ActivityType.DOORBELL_MOTION} ) if latest is None: return False return _activity_time_based_state(latest) def _retrieve_ding_state(data, detail): latest = data.activity_stream.get_latest_device_activity( detail.device_id, {ActivityType.DOORBELL_DING} ) if latest is None: return False if ( data.activity_stream.pubnub.connected and latest.action == ACTION_DOORBELL_CALL_MISSED ): return False return _activity_time_based_state(latest) def _activity_time_based_state(latest): """Get the latest state of the sensor.""" start = latest.activity_start_time end = latest.activity_end_time + TIME_TO_DECLARE_DETECTION return start <= _native_datetime() <= end def _native_datetime(): """Return time in the format august uses without timezone.""" return datetime.now() SENSOR_NAME = 0 SENSOR_DEVICE_CLASS = 1 SENSOR_STATE_PROVIDER = 2 SENSOR_STATE_IS_TIME_BASED = 3 # sensor_type: [name, device_class, state_provider, is_time_based] SENSOR_TYPES_DOORBELL = { "doorbell_ding": ["Ding", DEVICE_CLASS_OCCUPANCY, _retrieve_ding_state, True], "doorbell_motion": ["Motion", DEVICE_CLASS_MOTION, _retrieve_motion_state, True], "doorbell_online": [ "Online", DEVICE_CLASS_CONNECTIVITY, _retrieve_online_state, False, ], } async def async_setup_entry(hass, config_entry, async_add_entities): """Set up the August binary sensors.""" data = hass.data[DOMAIN][config_entry.entry_id][DATA_AUGUST] entities = [] for door in data.locks: detail = data.get_device_detail(door.device_id) if not detail.doorsense: _LOGGER.debug( "Not adding sensor class door for lock %s because it does not have doorsense", door.device_name, ) continue _LOGGER.debug("Adding sensor class door for %s", door.device_name) entities.append(AugustDoorBinarySensor(data, "door_open", door)) for doorbell in data.doorbells: for sensor_type, sensor in SENSOR_TYPES_DOORBELL.items(): _LOGGER.debug( "Adding doorbell sensor class %s for %s", sensor[SENSOR_DEVICE_CLASS], doorbell.device_name, ) entities.append(AugustDoorbellBinarySensor(data, sensor_type, doorbell)) async_add_entities(entities) class AugustDoorBinarySensor(AugustEntityMixin, BinarySensorEntity): """Representation of an August Door binary sensor.""" _attr_device_class = DEVICE_CLASS_DOOR def __init__(self, data, sensor_type, device): """Initialize the sensor.""" super().__init__(data, device) self._data = data self._sensor_type = sensor_type self._device = device self._attr_name = f"{device.device_name} Open" self._attr_unique_id = f"{self._device_id}_open" self._update_from_data() @callback def _update_from_data(self): """Get the latest state of the sensor and update activity.""" door_activity = self._data.activity_stream.get_latest_device_activity( self._device_id, {ActivityType.DOOR_OPERATION} ) if door_activity is not None: update_lock_detail_from_activity(self._detail, door_activity) # If the source is pubnub the lock must be online since its a live update if door_activity.source == SOURCE_PUBNUB: self._detail.set_online(True) bridge_activity = self._data.activity_stream.get_latest_device_activity( self._device_id, {ActivityType.BRIDGE_OPERATION} ) if bridge_activity is not None: update_lock_detail_from_activity(self._detail, bridge_activity) self._attr_available = self._detail.bridge_is_online self._attr_is_on = self._detail.door_state == LockDoorStatus.OPEN class AugustDoorbellBinarySensor(AugustEntityMixin, BinarySensorEntity): """Representation of an August binary sensor.""" def __init__(self, data, sensor_type, device): """Initialize the sensor.""" super().__init__(data, device) self._check_for_off_update_listener = None self._data = data self._sensor_type = sensor_type self._attr_device_class = self._sensor_config[SENSOR_DEVICE_CLASS] self._attr_name = f"{device.device_name} {self._sensor_config[SENSOR_NAME]}" self._attr_unique_id = ( f"{self._device_id}_{self._sensor_config[SENSOR_NAME].lower()}" ) self._update_from_data() @property def _sensor_config(self): """Return the config for the sensor.""" return SENSOR_TYPES_DOORBELL[self._sensor_type] @property def _state_provider(self): """Return the state provider for the binary sensor.""" return self._sensor_config[SENSOR_STATE_PROVIDER] @property def _is_time_based(self): """Return true of false if the sensor is time based.""" return self._sensor_config[SENSOR_STATE_IS_TIME_BASED] @callback def _update_from_data(self): """Get the latest state of the sensor.""" self._cancel_any_pending_updates() self._attr_is_on = self._state_provider(self._data, self._detail) if self._is_time_based: self._attr_available = _retrieve_online_state(self._data, self._detail) self._schedule_update_to_recheck_turn_off_sensor() else: self._attr_available = True def _schedule_update_to_recheck_turn_off_sensor(self): """Schedule an update to recheck the sensor to see if it is ready to turn off.""" # If the sensor is already off there is nothing to do if not self.is_on: return # self.hass is only available after setup is completed # and we will recheck in async_added_to_hass if not self.hass: return @callback def _scheduled_update(now): """Timer callback for sensor update.""" self._check_for_off_update_listener = None self._update_from_data() if not self.is_on: self.async_write_ha_state() self._check_for_off_update_listener = async_call_later( self.hass, TIME_TO_RECHECK_DETECTION.total_seconds(), _scheduled_update ) def _cancel_any_pending_updates(self): """Cancel any updates to recheck a sensor to see if it is ready to turn off.""" if not self._check_for_off_update_listener: return _LOGGER.debug("%s: canceled pending update", self.entity_id) self._check_for_off_update_listener() self._check_for_off_update_listener = None async def async_added_to_hass(self): """Call the mixin to subscribe and setup an async_track_point_in_utc_time to turn off the sensor if needed.""" self._schedule_update_to_recheck_turn_off_sensor() await super().async_added_to_hass()