2018-02-19 22:46:22 +00:00
|
|
|
"""Extend the basic Accessory and Bridge functions."""
|
2018-04-06 21:11:53 +00:00
|
|
|
from datetime import timedelta
|
2018-05-30 10:39:27 +00:00
|
|
|
from functools import partial, wraps
|
2018-04-06 21:11:53 +00:00
|
|
|
from inspect import getmodule
|
2018-02-26 03:27:40 +00:00
|
|
|
import logging
|
|
|
|
|
2018-05-04 14:46:00 +00:00
|
|
|
from pyhap.accessory import Accessory, Bridge
|
2018-03-15 01:48:21 +00:00
|
|
|
from pyhap.accessory_driver import AccessoryDriver
|
2018-05-04 14:46:00 +00:00
|
|
|
from pyhap.const import CATEGORY_OTHER
|
2018-02-19 22:46:22 +00:00
|
|
|
|
2020-05-01 04:05:06 +00:00
|
|
|
from homeassistant.components import cover, vacuum
|
|
|
|
from homeassistant.components.cover import DEVICE_CLASS_GARAGE, DEVICE_CLASS_GATE
|
|
|
|
from homeassistant.components.media_player import DEVICE_CLASS_TV
|
2018-06-17 18:54:34 +00:00
|
|
|
from homeassistant.const import (
|
2019-07-31 19:25:30 +00:00
|
|
|
ATTR_BATTERY_CHARGING,
|
|
|
|
ATTR_BATTERY_LEVEL,
|
2020-05-01 04:05:06 +00:00
|
|
|
ATTR_DEVICE_CLASS,
|
2019-07-31 19:25:30 +00:00
|
|
|
ATTR_ENTITY_ID,
|
|
|
|
ATTR_SERVICE,
|
2020-05-01 04:05:06 +00:00
|
|
|
ATTR_SUPPORTED_FEATURES,
|
|
|
|
ATTR_UNIT_OF_MEASUREMENT,
|
|
|
|
CONF_NAME,
|
|
|
|
CONF_TYPE,
|
|
|
|
DEVICE_CLASS_HUMIDITY,
|
|
|
|
DEVICE_CLASS_ILLUMINANCE,
|
|
|
|
DEVICE_CLASS_TEMPERATURE,
|
2020-04-22 00:43:49 +00:00
|
|
|
STATE_ON,
|
2020-05-01 04:05:06 +00:00
|
|
|
TEMP_CELSIUS,
|
|
|
|
TEMP_FAHRENHEIT,
|
|
|
|
UNIT_PERCENTAGE,
|
2019-07-31 19:25:30 +00:00
|
|
|
__version__,
|
|
|
|
)
|
2019-02-14 15:01:46 +00:00
|
|
|
from homeassistant.core import callback as ha_callback, split_entity_id
|
2018-04-06 21:11:53 +00:00
|
|
|
from homeassistant.helpers.event import (
|
2019-07-31 19:25:30 +00:00
|
|
|
async_track_state_change,
|
|
|
|
track_point_in_utc_time,
|
|
|
|
)
|
2018-04-06 21:11:53 +00:00
|
|
|
from homeassistant.util import dt as dt_util
|
2020-05-01 04:05:06 +00:00
|
|
|
from homeassistant.util.decorator import Registry
|
2018-03-16 00:05:28 +00:00
|
|
|
|
2018-02-19 22:46:22 +00:00
|
|
|
from .const import (
|
2019-07-31 19:25:30 +00:00
|
|
|
ATTR_DISPLAY_NAME,
|
|
|
|
ATTR_VALUE,
|
|
|
|
BRIDGE_MODEL,
|
|
|
|
BRIDGE_SERIAL_NUMBER,
|
|
|
|
CHAR_BATTERY_LEVEL,
|
|
|
|
CHAR_CHARGING_STATE,
|
|
|
|
CHAR_STATUS_LOW_BATTERY,
|
2020-05-01 04:05:06 +00:00
|
|
|
CONF_FEATURE_LIST,
|
2020-04-22 00:43:49 +00:00
|
|
|
CONF_LINKED_BATTERY_CHARGING_SENSOR,
|
2019-07-31 19:25:30 +00:00
|
|
|
CONF_LINKED_BATTERY_SENSOR,
|
|
|
|
CONF_LOW_BATTERY_THRESHOLD,
|
|
|
|
DEBOUNCE_TIMEOUT,
|
|
|
|
DEFAULT_LOW_BATTERY_THRESHOLD,
|
2020-05-01 04:05:06 +00:00
|
|
|
DEVICE_CLASS_CO,
|
|
|
|
DEVICE_CLASS_CO2,
|
|
|
|
DEVICE_CLASS_PM25,
|
2019-07-31 19:25:30 +00:00
|
|
|
EVENT_HOMEKIT_CHANGED,
|
2020-04-22 00:43:49 +00:00
|
|
|
HK_CHARGING,
|
|
|
|
HK_NOT_CHARGABLE,
|
|
|
|
HK_NOT_CHARGING,
|
2019-07-31 19:25:30 +00:00
|
|
|
MANUFACTURER,
|
|
|
|
SERV_BATTERY_SERVICE,
|
2020-05-01 04:05:06 +00:00
|
|
|
TYPE_FAUCET,
|
|
|
|
TYPE_OUTLET,
|
|
|
|
TYPE_SHOWER,
|
|
|
|
TYPE_SPRINKLER,
|
|
|
|
TYPE_SWITCH,
|
|
|
|
TYPE_VALVE,
|
|
|
|
)
|
|
|
|
from .util import (
|
|
|
|
convert_to_float,
|
|
|
|
dismiss_setup_message,
|
|
|
|
show_setup_message,
|
|
|
|
validate_media_player_features,
|
2019-07-31 19:25:30 +00:00
|
|
|
)
|
2018-02-19 22:46:22 +00:00
|
|
|
|
2018-02-26 03:27:40 +00:00
|
|
|
_LOGGER = logging.getLogger(__name__)
|
2020-05-01 04:05:06 +00:00
|
|
|
SWITCH_TYPES = {
|
|
|
|
TYPE_FAUCET: "Valve",
|
|
|
|
TYPE_OUTLET: "Outlet",
|
|
|
|
TYPE_SHOWER: "Valve",
|
|
|
|
TYPE_SPRINKLER: "Valve",
|
|
|
|
TYPE_SWITCH: "Switch",
|
|
|
|
TYPE_VALVE: "Valve",
|
|
|
|
}
|
|
|
|
TYPES = Registry()
|
2018-02-19 22:46:22 +00:00
|
|
|
|
|
|
|
|
2018-04-06 21:11:53 +00:00
|
|
|
def debounce(func):
|
2018-08-24 08:28:43 +00:00
|
|
|
"""Decorate function to debounce callbacks from HomeKit."""
|
2019-07-31 19:25:30 +00:00
|
|
|
|
2018-04-11 20:24:14 +00:00
|
|
|
@ha_callback
|
2018-05-30 10:39:27 +00:00
|
|
|
def call_later_listener(self, *args):
|
2018-08-24 08:28:43 +00:00
|
|
|
"""Handle call_later callback."""
|
2018-05-30 10:39:27 +00:00
|
|
|
debounce_params = self.debounce.pop(func.__name__, None)
|
|
|
|
if debounce_params:
|
2018-10-19 22:14:05 +00:00
|
|
|
self.hass.async_add_executor_job(func, self, *debounce_params[1:])
|
2018-04-06 21:11:53 +00:00
|
|
|
|
|
|
|
@wraps(func)
|
2018-05-30 10:39:27 +00:00
|
|
|
def wrapper(self, *args):
|
2018-08-24 08:28:43 +00:00
|
|
|
"""Start async timer."""
|
2018-05-30 10:39:27 +00:00
|
|
|
debounce_params = self.debounce.pop(func.__name__, None)
|
|
|
|
if debounce_params:
|
|
|
|
debounce_params[0]() # remove listener
|
2018-04-06 21:11:53 +00:00
|
|
|
remove_listener = track_point_in_utc_time(
|
2019-07-31 19:25:30 +00:00
|
|
|
self.hass,
|
|
|
|
partial(call_later_listener, self),
|
|
|
|
dt_util.utcnow() + timedelta(seconds=DEBOUNCE_TIMEOUT),
|
|
|
|
)
|
2018-05-30 10:39:27 +00:00
|
|
|
self.debounce[func.__name__] = (remove_listener, *args)
|
2019-07-31 19:25:30 +00:00
|
|
|
logger.debug(
|
|
|
|
"%s: Start %s timeout", self.entity_id, func.__name__.replace("set_", "")
|
|
|
|
)
|
2018-04-06 21:11:53 +00:00
|
|
|
|
|
|
|
name = getmodule(func).__name__
|
|
|
|
logger = logging.getLogger(name)
|
|
|
|
return wrapper
|
|
|
|
|
|
|
|
|
2020-05-01 04:05:06 +00:00
|
|
|
def get_accessory(hass, driver, state, aid, config):
|
|
|
|
"""Take state and return an accessory object if supported."""
|
|
|
|
if not aid:
|
|
|
|
_LOGGER.warning(
|
|
|
|
'The entity "%s" is not supported, since it '
|
|
|
|
"generates an invalid aid, please change it.",
|
|
|
|
state.entity_id,
|
|
|
|
)
|
|
|
|
return None
|
|
|
|
|
|
|
|
a_type = None
|
|
|
|
name = config.get(CONF_NAME, state.name)
|
|
|
|
|
|
|
|
if state.domain == "alarm_control_panel":
|
|
|
|
a_type = "SecuritySystem"
|
|
|
|
|
|
|
|
elif state.domain in ("binary_sensor", "device_tracker", "person"):
|
|
|
|
a_type = "BinarySensor"
|
|
|
|
|
|
|
|
elif state.domain == "climate":
|
|
|
|
a_type = "Thermostat"
|
|
|
|
|
|
|
|
elif state.domain == "cover":
|
|
|
|
device_class = state.attributes.get(ATTR_DEVICE_CLASS)
|
|
|
|
features = state.attributes.get(ATTR_SUPPORTED_FEATURES, 0)
|
|
|
|
|
|
|
|
if device_class in (DEVICE_CLASS_GARAGE, DEVICE_CLASS_GATE) and features & (
|
|
|
|
cover.SUPPORT_OPEN | cover.SUPPORT_CLOSE
|
|
|
|
):
|
|
|
|
a_type = "GarageDoorOpener"
|
|
|
|
elif features & cover.SUPPORT_SET_POSITION:
|
|
|
|
a_type = "WindowCovering"
|
|
|
|
elif features & (cover.SUPPORT_OPEN | cover.SUPPORT_CLOSE):
|
|
|
|
a_type = "WindowCoveringBasic"
|
|
|
|
|
|
|
|
elif state.domain == "fan":
|
|
|
|
a_type = "Fan"
|
|
|
|
|
|
|
|
elif state.domain == "light":
|
|
|
|
a_type = "Light"
|
|
|
|
|
|
|
|
elif state.domain == "lock":
|
|
|
|
a_type = "Lock"
|
|
|
|
|
|
|
|
elif state.domain == "media_player":
|
|
|
|
device_class = state.attributes.get(ATTR_DEVICE_CLASS)
|
|
|
|
feature_list = config.get(CONF_FEATURE_LIST)
|
|
|
|
|
|
|
|
if device_class == DEVICE_CLASS_TV:
|
|
|
|
a_type = "TelevisionMediaPlayer"
|
|
|
|
else:
|
|
|
|
if feature_list and validate_media_player_features(state, feature_list):
|
|
|
|
a_type = "MediaPlayer"
|
|
|
|
|
|
|
|
elif state.domain == "sensor":
|
|
|
|
device_class = state.attributes.get(ATTR_DEVICE_CLASS)
|
|
|
|
unit = state.attributes.get(ATTR_UNIT_OF_MEASUREMENT)
|
|
|
|
|
|
|
|
if device_class == DEVICE_CLASS_TEMPERATURE or unit in (
|
|
|
|
TEMP_CELSIUS,
|
|
|
|
TEMP_FAHRENHEIT,
|
|
|
|
):
|
|
|
|
a_type = "TemperatureSensor"
|
|
|
|
elif device_class == DEVICE_CLASS_HUMIDITY and unit == UNIT_PERCENTAGE:
|
|
|
|
a_type = "HumiditySensor"
|
|
|
|
elif device_class == DEVICE_CLASS_PM25 or DEVICE_CLASS_PM25 in state.entity_id:
|
|
|
|
a_type = "AirQualitySensor"
|
|
|
|
elif device_class == DEVICE_CLASS_CO:
|
|
|
|
a_type = "CarbonMonoxideSensor"
|
|
|
|
elif device_class == DEVICE_CLASS_CO2 or DEVICE_CLASS_CO2 in state.entity_id:
|
|
|
|
a_type = "CarbonDioxideSensor"
|
|
|
|
elif device_class == DEVICE_CLASS_ILLUMINANCE or unit in ("lm", "lx"):
|
|
|
|
a_type = "LightSensor"
|
|
|
|
|
|
|
|
elif state.domain == "switch":
|
|
|
|
switch_type = config.get(CONF_TYPE, TYPE_SWITCH)
|
|
|
|
a_type = SWITCH_TYPES[switch_type]
|
|
|
|
|
|
|
|
elif state.domain == "vacuum":
|
|
|
|
features = state.attributes.get(ATTR_SUPPORTED_FEATURES, 0)
|
|
|
|
if features & (vacuum.SUPPORT_START | vacuum.SUPPORT_RETURN_HOME):
|
|
|
|
a_type = "DockVacuum"
|
|
|
|
else:
|
|
|
|
a_type = "Switch"
|
|
|
|
|
|
|
|
elif state.domain in ("automation", "input_boolean", "remote", "scene", "script"):
|
|
|
|
a_type = "Switch"
|
|
|
|
|
|
|
|
elif state.domain == "water_heater":
|
|
|
|
a_type = "WaterHeater"
|
|
|
|
|
|
|
|
if a_type is None:
|
|
|
|
return None
|
|
|
|
|
|
|
|
_LOGGER.debug('Add "%s" as "%s"', state.entity_id, a_type)
|
|
|
|
return TYPES[a_type](hass, driver, name, state.entity_id, aid, config)
|
|
|
|
|
|
|
|
|
2018-02-26 03:27:40 +00:00
|
|
|
class HomeAccessory(Accessory):
|
2018-03-15 01:48:21 +00:00
|
|
|
"""Adapter class for Accessory."""
|
2018-02-26 03:27:40 +00:00
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
def __init__(
|
|
|
|
self, hass, driver, name, entity_id, aid, config, category=CATEGORY_OTHER
|
|
|
|
):
|
2018-02-26 03:27:40 +00:00
|
|
|
"""Initialize a Accessory object."""
|
2018-05-29 20:43:26 +00:00
|
|
|
super().__init__(driver, name, aid=aid)
|
2018-05-11 12:22:45 +00:00
|
|
|
model = split_entity_id(entity_id)[0].replace("_", " ").title()
|
2018-05-04 14:46:00 +00:00
|
|
|
self.set_info_service(
|
2019-07-31 19:25:30 +00:00
|
|
|
firmware_revision=__version__,
|
|
|
|
manufacturer=MANUFACTURER,
|
|
|
|
model=model,
|
|
|
|
serial_number=entity_id,
|
|
|
|
)
|
2018-05-04 14:46:00 +00:00
|
|
|
self.category = category
|
2019-04-09 21:13:48 +00:00
|
|
|
self.config = config or {}
|
2018-04-11 20:24:14 +00:00
|
|
|
self.entity_id = entity_id
|
|
|
|
self.hass = hass
|
2018-05-30 10:39:27 +00:00
|
|
|
self.debounce = {}
|
2020-04-30 07:08:56 +00:00
|
|
|
self._char_battery = None
|
|
|
|
self._char_charging = None
|
|
|
|
self._char_low_battery = None
|
2019-07-31 19:25:30 +00:00
|
|
|
self.linked_battery_sensor = self.config.get(CONF_LINKED_BATTERY_SENSOR)
|
2020-04-22 00:43:49 +00:00
|
|
|
self.linked_battery_charging_sensor = self.config.get(
|
|
|
|
CONF_LINKED_BATTERY_CHARGING_SENSOR
|
|
|
|
)
|
2019-07-31 19:25:30 +00:00
|
|
|
self.low_battery_threshold = self.config.get(
|
|
|
|
CONF_LOW_BATTERY_THRESHOLD, DEFAULT_LOW_BATTERY_THRESHOLD
|
|
|
|
)
|
2018-06-17 18:54:34 +00:00
|
|
|
|
|
|
|
"""Add battery service if available"""
|
2020-04-22 00:43:49 +00:00
|
|
|
entity_attributes = self.hass.states.get(self.entity_id).attributes
|
|
|
|
battery_found = entity_attributes.get(ATTR_BATTERY_LEVEL)
|
2019-11-26 16:55:33 +00:00
|
|
|
|
2019-04-09 21:13:48 +00:00
|
|
|
if self.linked_battery_sensor:
|
2019-11-26 16:55:33 +00:00
|
|
|
state = self.hass.states.get(self.linked_battery_sensor)
|
|
|
|
if state is not None:
|
|
|
|
battery_found = state.state
|
|
|
|
else:
|
|
|
|
self.linked_battery_sensor = None
|
|
|
|
_LOGGER.warning(
|
|
|
|
"%s: Battery sensor state missing: %s",
|
|
|
|
self.entity_id,
|
|
|
|
self.linked_battery_sensor,
|
|
|
|
)
|
2019-04-09 21:13:48 +00:00
|
|
|
|
2020-04-22 00:43:49 +00:00
|
|
|
if not battery_found:
|
2018-06-17 18:54:34 +00:00
|
|
|
return
|
2020-04-22 00:43:49 +00:00
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
_LOGGER.debug("%s: Found battery level", self.entity_id)
|
2020-04-22 00:43:49 +00:00
|
|
|
|
|
|
|
if self.linked_battery_charging_sensor:
|
|
|
|
state = self.hass.states.get(self.linked_battery_charging_sensor)
|
|
|
|
if state is None:
|
|
|
|
self.linked_battery_charging_sensor = None
|
|
|
|
_LOGGER.warning(
|
|
|
|
"%s: Battery charging binary_sensor state missing: %s",
|
|
|
|
self.entity_id,
|
|
|
|
self.linked_battery_charging_sensor,
|
|
|
|
)
|
|
|
|
else:
|
|
|
|
_LOGGER.debug("%s: Found battery charging", self.entity_id)
|
|
|
|
|
2018-06-17 18:54:34 +00:00
|
|
|
serv_battery = self.add_preload_service(SERV_BATTERY_SERVICE)
|
2019-07-31 19:25:30 +00:00
|
|
|
self._char_battery = serv_battery.configure_char(CHAR_BATTERY_LEVEL, value=0)
|
2020-04-22 00:43:49 +00:00
|
|
|
self._char_charging = serv_battery.configure_char(
|
|
|
|
CHAR_CHARGING_STATE, value=HK_NOT_CHARGABLE
|
|
|
|
)
|
2018-06-17 18:54:34 +00:00
|
|
|
self._char_low_battery = serv_battery.configure_char(
|
2019-07-31 19:25:30 +00:00
|
|
|
CHAR_STATUS_LOW_BATTERY, value=0
|
|
|
|
)
|
2018-02-19 22:46:22 +00:00
|
|
|
|
2018-05-30 10:39:27 +00:00
|
|
|
async def run(self):
|
2018-08-24 08:28:43 +00:00
|
|
|
"""Handle accessory driver started event.
|
2018-05-30 10:39:27 +00:00
|
|
|
|
|
|
|
Run inside the HAP-python event loop.
|
|
|
|
"""
|
2018-10-19 22:14:05 +00:00
|
|
|
self.hass.add_job(self.run_handler)
|
|
|
|
|
|
|
|
async def run_handler(self):
|
|
|
|
"""Handle accessory driver started event.
|
|
|
|
|
|
|
|
Run inside the Home Assistant event loop.
|
|
|
|
"""
|
2018-04-04 22:52:25 +00:00
|
|
|
state = self.hass.states.get(self.entity_id)
|
2020-04-30 22:49:16 +00:00
|
|
|
await self.async_update_state_callback(None, None, state)
|
|
|
|
async_track_state_change(
|
|
|
|
self.hass, self.entity_id, self.async_update_state_callback
|
|
|
|
)
|
2018-04-11 20:24:14 +00:00
|
|
|
|
2020-04-22 00:43:49 +00:00
|
|
|
battery_charging_state = None
|
|
|
|
battery_state = None
|
2019-04-09 21:13:48 +00:00
|
|
|
if self.linked_battery_sensor:
|
2020-04-22 00:43:49 +00:00
|
|
|
linked_battery_sensor_state = self.hass.states.get(
|
|
|
|
self.linked_battery_sensor
|
|
|
|
)
|
|
|
|
battery_state = linked_battery_sensor_state.state
|
|
|
|
battery_charging_state = linked_battery_sensor_state.attributes.get(
|
|
|
|
ATTR_BATTERY_CHARGING
|
2019-07-31 19:25:30 +00:00
|
|
|
)
|
2019-04-09 21:13:48 +00:00
|
|
|
async_track_state_change(
|
2020-04-30 22:49:16 +00:00
|
|
|
self.hass, self.linked_battery_sensor, self.async_update_linked_battery
|
2019-07-31 19:25:30 +00:00
|
|
|
)
|
2020-04-22 00:43:49 +00:00
|
|
|
else:
|
|
|
|
battery_state = state.attributes.get(ATTR_BATTERY_LEVEL)
|
|
|
|
if self.linked_battery_charging_sensor:
|
|
|
|
battery_charging_state = (
|
|
|
|
self.hass.states.get(self.linked_battery_charging_sensor).state
|
|
|
|
== STATE_ON
|
|
|
|
)
|
|
|
|
async_track_state_change(
|
|
|
|
self.hass,
|
|
|
|
self.linked_battery_charging_sensor,
|
2020-04-30 22:49:16 +00:00
|
|
|
self.async_update_linked_battery_charging,
|
2020-04-22 00:43:49 +00:00
|
|
|
)
|
|
|
|
elif battery_charging_state is None:
|
|
|
|
battery_charging_state = state.attributes.get(ATTR_BATTERY_CHARGING)
|
|
|
|
|
|
|
|
if battery_state is not None or battery_charging_state is not None:
|
|
|
|
self.hass.async_add_executor_job(
|
|
|
|
self.update_battery, battery_state, battery_charging_state
|
|
|
|
)
|
2019-04-09 21:13:48 +00:00
|
|
|
|
2020-04-30 22:49:16 +00:00
|
|
|
async def async_update_state_callback(
|
|
|
|
self, entity_id=None, old_state=None, new_state=None
|
|
|
|
):
|
2018-08-24 08:28:43 +00:00
|
|
|
"""Handle state change listener callback."""
|
2019-07-31 19:25:30 +00:00
|
|
|
_LOGGER.debug("New_state: %s", new_state)
|
2018-04-11 20:24:14 +00:00
|
|
|
if new_state is None:
|
|
|
|
return
|
2020-04-22 00:43:49 +00:00
|
|
|
battery_state = None
|
|
|
|
battery_charging_state = None
|
|
|
|
if (
|
|
|
|
not self.linked_battery_sensor
|
|
|
|
and ATTR_BATTERY_LEVEL in new_state.attributes
|
|
|
|
):
|
|
|
|
battery_state = new_state.attributes.get(ATTR_BATTERY_LEVEL)
|
|
|
|
if (
|
|
|
|
not self.linked_battery_charging_sensor
|
|
|
|
and ATTR_BATTERY_CHARGING in new_state.attributes
|
|
|
|
):
|
|
|
|
battery_charging_state = new_state.attributes.get(ATTR_BATTERY_CHARGING)
|
|
|
|
if battery_state is not None or battery_charging_state is not None:
|
2020-04-30 22:49:16 +00:00
|
|
|
await self.hass.async_add_executor_job(
|
2020-04-22 00:43:49 +00:00
|
|
|
self.update_battery, battery_state, battery_charging_state
|
|
|
|
)
|
2020-04-30 22:49:16 +00:00
|
|
|
await self.hass.async_add_executor_job(self.update_state, new_state)
|
2018-04-11 20:24:14 +00:00
|
|
|
|
2020-04-30 22:49:16 +00:00
|
|
|
async def async_update_linked_battery(
|
|
|
|
self, entity_id=None, old_state=None, new_state=None
|
|
|
|
):
|
2019-04-09 21:13:48 +00:00
|
|
|
"""Handle linked battery sensor state change listener callback."""
|
2020-04-22 00:43:49 +00:00
|
|
|
if self.linked_battery_charging_sensor:
|
|
|
|
battery_charging_state = None
|
|
|
|
else:
|
|
|
|
battery_charging_state = new_state.attributes.get(ATTR_BATTERY_CHARGING)
|
2020-04-30 22:49:16 +00:00
|
|
|
await self.hass.async_add_executor_job(
|
2020-04-22 00:43:49 +00:00
|
|
|
self.update_battery, new_state.state, battery_charging_state,
|
|
|
|
)
|
|
|
|
|
2020-04-30 22:49:16 +00:00
|
|
|
async def async_update_linked_battery_charging(
|
2020-04-22 00:43:49 +00:00
|
|
|
self, entity_id=None, old_state=None, new_state=None
|
|
|
|
):
|
|
|
|
"""Handle linked battery charging sensor state change listener callback."""
|
2020-04-30 22:49:16 +00:00
|
|
|
await self.hass.async_add_executor_job(
|
2020-04-22 00:43:49 +00:00
|
|
|
self.update_battery, None, new_state.state == STATE_ON
|
|
|
|
)
|
2019-04-09 21:13:48 +00:00
|
|
|
|
2020-04-22 00:43:49 +00:00
|
|
|
def update_battery(self, battery_level, battery_charging):
|
2018-06-17 18:54:34 +00:00
|
|
|
"""Update battery service if available.
|
|
|
|
|
|
|
|
Only call this function if self._support_battery_level is True.
|
|
|
|
"""
|
2020-04-30 07:08:56 +00:00
|
|
|
if not self._char_battery:
|
|
|
|
# Battery appeared after homekit was started
|
|
|
|
return
|
|
|
|
|
2020-04-22 00:43:49 +00:00
|
|
|
battery_level = convert_to_float(battery_level)
|
|
|
|
if battery_level is not None:
|
|
|
|
if self._char_battery.value != battery_level:
|
|
|
|
self._char_battery.set_value(battery_level)
|
|
|
|
is_low_battery = 1 if battery_level < self.low_battery_threshold else 0
|
|
|
|
if self._char_low_battery.value != is_low_battery:
|
|
|
|
self._char_low_battery.set_value(is_low_battery)
|
|
|
|
_LOGGER.debug(
|
|
|
|
"%s: Updated battery level to %d", self.entity_id, battery_level
|
|
|
|
)
|
|
|
|
|
2020-04-30 07:08:56 +00:00
|
|
|
# Charging state can appear after homekit was started
|
|
|
|
if battery_charging is None or not self._char_charging:
|
2018-06-17 18:54:34 +00:00
|
|
|
return
|
2020-04-22 00:43:49 +00:00
|
|
|
|
|
|
|
hk_charging = HK_CHARGING if battery_charging else HK_NOT_CHARGING
|
|
|
|
if self._char_charging.value != hk_charging:
|
|
|
|
self._char_charging.set_value(hk_charging)
|
|
|
|
_LOGGER.debug(
|
|
|
|
"%s: Updated battery charging to %d", self.entity_id, hk_charging
|
|
|
|
)
|
2018-06-17 18:54:34 +00:00
|
|
|
|
2018-04-11 20:24:14 +00:00
|
|
|
def update_state(self, new_state):
|
2018-08-24 08:28:43 +00:00
|
|
|
"""Handle state change to update HomeKit value.
|
2018-04-11 20:24:14 +00:00
|
|
|
|
|
|
|
Overridden by accessory types.
|
|
|
|
"""
|
2018-05-21 02:25:53 +00:00
|
|
|
raise NotImplementedError()
|
2018-03-16 00:05:28 +00:00
|
|
|
|
2018-10-16 11:32:53 +00:00
|
|
|
def call_service(self, domain, service, service_data, value=None):
|
|
|
|
"""Fire event and call service for changes from HomeKit."""
|
2019-07-31 19:25:30 +00:00
|
|
|
self.hass.add_job(self.async_call_service, domain, service, service_data, value)
|
2018-10-16 11:32:53 +00:00
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
async def async_call_service(self, domain, service, service_data, value=None):
|
2018-10-16 11:32:53 +00:00
|
|
|
"""Fire event and call service for changes from HomeKit.
|
|
|
|
|
|
|
|
This method must be run in the event loop.
|
|
|
|
"""
|
|
|
|
event_data = {
|
|
|
|
ATTR_ENTITY_ID: self.entity_id,
|
|
|
|
ATTR_DISPLAY_NAME: self.display_name,
|
|
|
|
ATTR_SERVICE: service,
|
2019-07-31 19:25:30 +00:00
|
|
|
ATTR_VALUE: value,
|
2018-10-16 11:32:53 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
self.hass.bus.async_fire(EVENT_HOMEKIT_CHANGED, event_data)
|
|
|
|
await self.hass.services.async_call(domain, service, service_data)
|
|
|
|
|
2018-02-19 22:46:22 +00:00
|
|
|
|
|
|
|
class HomeBridge(Bridge):
|
2018-03-15 01:48:21 +00:00
|
|
|
"""Adapter class for Bridge."""
|
2018-02-19 22:46:22 +00:00
|
|
|
|
2018-07-22 07:51:42 +00:00
|
|
|
def __init__(self, hass, driver, name):
|
2018-02-19 22:46:22 +00:00
|
|
|
"""Initialize a Bridge object."""
|
2018-05-29 20:43:26 +00:00
|
|
|
super().__init__(driver, name)
|
2018-05-04 14:46:00 +00:00
|
|
|
self.set_info_service(
|
2019-07-31 19:25:30 +00:00
|
|
|
firmware_revision=__version__,
|
|
|
|
manufacturer=MANUFACTURER,
|
|
|
|
model=BRIDGE_MODEL,
|
|
|
|
serial_number=BRIDGE_SERIAL_NUMBER,
|
|
|
|
)
|
2018-04-04 22:52:25 +00:00
|
|
|
self.hass = hass
|
2018-03-01 23:20:02 +00:00
|
|
|
|
2018-03-15 01:48:21 +00:00
|
|
|
def setup_message(self):
|
|
|
|
"""Prevent print of pyhap setup message to terminal."""
|
|
|
|
|
|
|
|
|
|
|
|
class HomeDriver(AccessoryDriver):
|
|
|
|
"""Adapter class for AccessoryDriver."""
|
|
|
|
|
2020-05-01 04:05:06 +00:00
|
|
|
def __init__(self, hass, entry_id, bridge_name, **kwargs):
|
2018-03-15 01:48:21 +00:00
|
|
|
"""Initialize a AccessoryDriver object."""
|
2018-05-29 20:43:26 +00:00
|
|
|
super().__init__(**kwargs)
|
2018-05-18 14:32:57 +00:00
|
|
|
self.hass = hass
|
2020-05-01 04:05:06 +00:00
|
|
|
self._entry_id = entry_id
|
|
|
|
self._bridge_name = bridge_name
|
2018-05-18 14:32:57 +00:00
|
|
|
|
|
|
|
def pair(self, client_uuid, client_public):
|
|
|
|
"""Override super function to dismiss setup message if paired."""
|
2018-05-30 10:39:27 +00:00
|
|
|
success = super().pair(client_uuid, client_public)
|
|
|
|
if success:
|
2020-05-01 04:05:06 +00:00
|
|
|
dismiss_setup_message(self.hass, self._entry_id)
|
2018-05-30 10:39:27 +00:00
|
|
|
return success
|
2018-05-18 14:32:57 +00:00
|
|
|
|
|
|
|
def unpair(self, client_uuid):
|
|
|
|
"""Override super function to show setup message if unpaired."""
|
|
|
|
super().unpair(client_uuid)
|
2020-05-01 04:05:06 +00:00
|
|
|
show_setup_message(
|
|
|
|
self.hass,
|
|
|
|
self._entry_id,
|
|
|
|
self._bridge_name,
|
|
|
|
self.state.pincode,
|
|
|
|
self.accessory.xhm_uri(),
|
|
|
|
)
|