2019-02-28 18:09:04 +00:00
|
|
|
"""Support for Homekit sensors."""
|
2021-10-03 23:42:25 +00:00
|
|
|
from __future__ import annotations
|
|
|
|
|
|
|
|
from collections.abc import Callable
|
|
|
|
from dataclasses import dataclass
|
|
|
|
|
2022-09-26 01:31:56 +00:00
|
|
|
from aiohomekit.model import Accessory, Transport
|
2021-06-28 21:48:29 +00:00
|
|
|
from aiohomekit.model.characteristics import Characteristic, CharacteristicsTypes
|
2022-08-04 10:55:29 +00:00
|
|
|
from aiohomekit.model.characteristics.const import ThreadNodeCapabilities, ThreadStatus
|
2022-02-01 19:30:37 +00:00
|
|
|
from aiohomekit.model.services import Service, ServicesTypes
|
2019-08-01 10:24:46 +00:00
|
|
|
|
2022-10-15 17:57:23 +00:00
|
|
|
from homeassistant.components.bluetooth import (
|
|
|
|
async_ble_device_from_address,
|
|
|
|
async_last_service_info,
|
|
|
|
)
|
2021-10-03 23:42:25 +00:00
|
|
|
from homeassistant.components.sensor import (
|
2021-12-13 16:04:21 +00:00
|
|
|
SensorDeviceClass,
|
2021-10-03 23:42:25 +00:00
|
|
|
SensorEntity,
|
|
|
|
SensorEntityDescription,
|
2021-12-13 16:04:21 +00:00
|
|
|
SensorStateClass,
|
2021-10-03 23:42:25 +00:00
|
|
|
)
|
2021-12-28 18:24:40 +00:00
|
|
|
from homeassistant.config_entries import ConfigEntry
|
2020-02-25 01:52:14 +00:00
|
|
|
from homeassistant.const import (
|
2021-08-17 14:29:52 +00:00
|
|
|
CONCENTRATION_MICROGRAMS_PER_CUBIC_METER,
|
2020-02-25 01:52:14 +00:00
|
|
|
CONCENTRATION_PARTS_PER_MILLION,
|
2021-12-23 19:05:36 +00:00
|
|
|
ELECTRIC_CURRENT_AMPERE,
|
2022-01-24 17:46:45 +00:00
|
|
|
ELECTRIC_POTENTIAL_VOLT,
|
2021-12-23 19:05:36 +00:00
|
|
|
ENERGY_KILO_WATT_HOUR,
|
2020-09-23 18:48:01 +00:00
|
|
|
LIGHT_LUX,
|
2020-09-05 19:09:14 +00:00
|
|
|
PERCENTAGE,
|
2021-08-02 17:47:11 +00:00
|
|
|
POWER_WATT,
|
2021-08-03 12:30:50 +00:00
|
|
|
PRESSURE_HPA,
|
2022-09-26 01:31:56 +00:00
|
|
|
SIGNAL_STRENGTH_DECIBELS_MILLIWATT,
|
2022-10-24 15:31:26 +00:00
|
|
|
SOUND_PRESSURE_DB,
|
2020-02-25 01:52:14 +00:00
|
|
|
TEMP_CELSIUS,
|
2022-10-11 21:26:03 +00:00
|
|
|
Platform,
|
2020-02-25 01:52:14 +00:00
|
|
|
)
|
2021-12-28 18:24:40 +00:00
|
|
|
from homeassistant.core import HomeAssistant, callback
|
2022-08-03 21:03:10 +00:00
|
|
|
from homeassistant.helpers.entity import EntityCategory
|
2021-12-28 18:24:40 +00:00
|
|
|
from homeassistant.helpers.entity_platform import AddEntitiesCallback
|
2022-02-01 19:30:37 +00:00
|
|
|
from homeassistant.helpers.typing import ConfigType
|
2019-02-28 18:09:04 +00:00
|
|
|
|
2022-08-06 18:10:26 +00:00
|
|
|
from . import KNOWN_DEVICES
|
2022-02-01 19:30:37 +00:00
|
|
|
from .connection import HKDevice
|
2022-08-06 18:10:26 +00:00
|
|
|
from .entity import CharacteristicEntity, HomeKitEntity
|
2022-07-03 20:47:54 +00:00
|
|
|
from .utils import folded_name
|
2019-03-21 05:56:46 +00:00
|
|
|
|
2019-02-28 18:09:04 +00:00
|
|
|
|
2021-10-03 23:42:25 +00:00
|
|
|
@dataclass
|
|
|
|
class HomeKitSensorEntityDescription(SensorEntityDescription):
|
|
|
|
"""Describes Homekit sensor."""
|
|
|
|
|
|
|
|
probe: Callable[[Characteristic], bool] | None = None
|
2022-08-03 21:03:10 +00:00
|
|
|
format: Callable[[Characteristic], str] | None = None
|
|
|
|
|
|
|
|
|
|
|
|
def thread_node_capability_to_str(char: Characteristic) -> str:
|
|
|
|
"""
|
|
|
|
Return the thread device type as a string.
|
|
|
|
|
|
|
|
The underlying value is a bitmask, but we want to turn that to
|
|
|
|
a human readable string. Some devices will have multiple capabilities.
|
|
|
|
For example, an NL55 is SLEEPY | MINIMAL. In that case we return the
|
|
|
|
"best" capability.
|
|
|
|
|
|
|
|
https://openthread.io/guides/thread-primer/node-roles-and-types
|
|
|
|
"""
|
|
|
|
|
|
|
|
val = ThreadNodeCapabilities(char.value)
|
|
|
|
|
|
|
|
if val & ThreadNodeCapabilities.BORDER_ROUTER_CAPABLE:
|
|
|
|
# can act as a bridge between thread network and e.g. WiFi
|
|
|
|
return "border_router_capable"
|
|
|
|
|
|
|
|
if val & ThreadNodeCapabilities.ROUTER_ELIGIBLE:
|
|
|
|
# radio always on, can be a router
|
|
|
|
return "router_eligible"
|
|
|
|
|
|
|
|
if val & ThreadNodeCapabilities.FULL:
|
|
|
|
# radio always on, but can't be a router
|
|
|
|
return "full"
|
|
|
|
|
|
|
|
if val & ThreadNodeCapabilities.MINIMAL:
|
|
|
|
# transceiver always on, does not need to poll for messages from its parent
|
|
|
|
return "minimal"
|
|
|
|
|
|
|
|
if val & ThreadNodeCapabilities.SLEEPY:
|
|
|
|
# normally disabled, wakes on occasion to poll for messages from its parent
|
|
|
|
return "sleepy"
|
|
|
|
|
|
|
|
# Device has no known thread capabilities
|
|
|
|
return "none"
|
2021-10-03 23:42:25 +00:00
|
|
|
|
|
|
|
|
2022-08-04 10:55:29 +00:00
|
|
|
def thread_status_to_str(char: Characteristic) -> str:
|
|
|
|
"""
|
|
|
|
Return the thread status as a string.
|
|
|
|
|
|
|
|
The underlying value is a bitmask, but we want to turn that to
|
|
|
|
a human readable string. So we check the flags in order. E.g. BORDER_ROUTER implies
|
|
|
|
ROUTER, so its more important to show that value.
|
|
|
|
"""
|
|
|
|
|
|
|
|
val = ThreadStatus(char.value)
|
|
|
|
|
|
|
|
if val & ThreadStatus.BORDER_ROUTER:
|
|
|
|
# Device has joined the Thread network and is participating
|
|
|
|
# in routing between mesh nodes.
|
|
|
|
# It's also the border router - bridging the thread network
|
|
|
|
# to WiFI/Ethernet/etc
|
|
|
|
return "border_router"
|
|
|
|
|
|
|
|
if val & ThreadStatus.LEADER:
|
|
|
|
# Device has joined the Thread network and is participating
|
|
|
|
# in routing between mesh nodes.
|
|
|
|
# It's also the leader. There's only one leader and it manages
|
|
|
|
# which nodes are routers.
|
|
|
|
return "leader"
|
|
|
|
|
|
|
|
if val & ThreadStatus.ROUTER:
|
|
|
|
# Device has joined the Thread network and is participating
|
|
|
|
# in routing between mesh nodes.
|
|
|
|
return "router"
|
|
|
|
|
|
|
|
if val & ThreadStatus.CHILD:
|
|
|
|
# Device has joined the Thread network as a child
|
|
|
|
# It's not participating in routing between mesh nodes
|
|
|
|
return "child"
|
|
|
|
|
|
|
|
if val & ThreadStatus.JOINING:
|
|
|
|
# Device is currently joining its Thread network
|
|
|
|
return "joining"
|
|
|
|
|
|
|
|
if val & ThreadStatus.DETACHED:
|
|
|
|
# Device is currently unable to reach its Thread network
|
|
|
|
return "detached"
|
|
|
|
|
|
|
|
# Must be ThreadStatus.DISABLED
|
|
|
|
# Device is not currently connected to Thread and will not try to.
|
|
|
|
return "disabled"
|
|
|
|
|
|
|
|
|
2021-10-03 23:42:25 +00:00
|
|
|
SIMPLE_SENSOR: dict[str, HomeKitSensorEntityDescription] = {
|
2022-01-31 22:48:16 +00:00
|
|
|
CharacteristicsTypes.VENDOR_CONNECTSENSE_ENERGY_WATT: HomeKitSensorEntityDescription(
|
|
|
|
key=CharacteristicsTypes.VENDOR_CONNECTSENSE_ENERGY_WATT,
|
2022-01-27 16:08:26 +00:00
|
|
|
name="Power",
|
2021-12-23 19:05:36 +00:00
|
|
|
device_class=SensorDeviceClass.POWER,
|
|
|
|
state_class=SensorStateClass.MEASUREMENT,
|
|
|
|
native_unit_of_measurement=POWER_WATT,
|
|
|
|
),
|
2022-01-31 22:48:16 +00:00
|
|
|
CharacteristicsTypes.VENDOR_CONNECTSENSE_ENERGY_AMPS: HomeKitSensorEntityDescription(
|
|
|
|
key=CharacteristicsTypes.VENDOR_CONNECTSENSE_ENERGY_AMPS,
|
2022-01-27 16:08:26 +00:00
|
|
|
name="Current",
|
2021-12-23 19:05:36 +00:00
|
|
|
device_class=SensorDeviceClass.CURRENT,
|
|
|
|
state_class=SensorStateClass.MEASUREMENT,
|
|
|
|
native_unit_of_measurement=ELECTRIC_CURRENT_AMPERE,
|
|
|
|
),
|
2022-01-31 22:48:16 +00:00
|
|
|
CharacteristicsTypes.VENDOR_CONNECTSENSE_ENERGY_AMPS_20: HomeKitSensorEntityDescription(
|
|
|
|
key=CharacteristicsTypes.VENDOR_CONNECTSENSE_ENERGY_AMPS_20,
|
2022-01-27 16:08:26 +00:00
|
|
|
name="Current",
|
2021-12-23 19:05:36 +00:00
|
|
|
device_class=SensorDeviceClass.CURRENT,
|
|
|
|
state_class=SensorStateClass.MEASUREMENT,
|
|
|
|
native_unit_of_measurement=ELECTRIC_CURRENT_AMPERE,
|
|
|
|
),
|
2022-01-31 22:48:16 +00:00
|
|
|
CharacteristicsTypes.VENDOR_CONNECTSENSE_ENERGY_KW_HOUR: HomeKitSensorEntityDescription(
|
|
|
|
key=CharacteristicsTypes.VENDOR_CONNECTSENSE_ENERGY_KW_HOUR,
|
2021-12-23 19:05:36 +00:00
|
|
|
name="Energy kWh",
|
|
|
|
device_class=SensorDeviceClass.ENERGY,
|
|
|
|
state_class=SensorStateClass.MEASUREMENT,
|
|
|
|
native_unit_of_measurement=ENERGY_KILO_WATT_HOUR,
|
|
|
|
),
|
2022-01-31 22:48:16 +00:00
|
|
|
CharacteristicsTypes.VENDOR_EVE_ENERGY_WATT: HomeKitSensorEntityDescription(
|
|
|
|
key=CharacteristicsTypes.VENDOR_EVE_ENERGY_WATT,
|
2022-01-27 16:08:26 +00:00
|
|
|
name="Power",
|
2021-12-13 16:04:21 +00:00
|
|
|
device_class=SensorDeviceClass.POWER,
|
|
|
|
state_class=SensorStateClass.MEASUREMENT,
|
2021-10-03 23:42:25 +00:00
|
|
|
native_unit_of_measurement=POWER_WATT,
|
|
|
|
),
|
2022-01-31 22:48:16 +00:00
|
|
|
CharacteristicsTypes.VENDOR_EVE_ENERGY_KW_HOUR: HomeKitSensorEntityDescription(
|
|
|
|
key=CharacteristicsTypes.VENDOR_EVE_ENERGY_KW_HOUR,
|
2022-01-24 17:46:45 +00:00
|
|
|
name="Energy kWh",
|
|
|
|
device_class=SensorDeviceClass.ENERGY,
|
2022-10-30 15:32:19 +00:00
|
|
|
state_class=SensorStateClass.TOTAL_INCREASING,
|
2022-01-24 17:46:45 +00:00
|
|
|
native_unit_of_measurement=ENERGY_KILO_WATT_HOUR,
|
|
|
|
),
|
2022-01-31 22:48:16 +00:00
|
|
|
CharacteristicsTypes.VENDOR_EVE_ENERGY_VOLTAGE: HomeKitSensorEntityDescription(
|
|
|
|
key=CharacteristicsTypes.VENDOR_EVE_ENERGY_VOLTAGE,
|
2022-01-24 17:46:45 +00:00
|
|
|
name="Volts",
|
|
|
|
device_class=SensorDeviceClass.VOLTAGE,
|
|
|
|
state_class=SensorStateClass.MEASUREMENT,
|
|
|
|
native_unit_of_measurement=ELECTRIC_POTENTIAL_VOLT,
|
|
|
|
),
|
2022-01-31 22:48:16 +00:00
|
|
|
CharacteristicsTypes.VENDOR_EVE_ENERGY_AMPERE: HomeKitSensorEntityDescription(
|
|
|
|
key=CharacteristicsTypes.VENDOR_EVE_ENERGY_AMPERE,
|
2022-01-24 17:46:45 +00:00
|
|
|
name="Amps",
|
|
|
|
device_class=SensorDeviceClass.CURRENT,
|
|
|
|
state_class=SensorStateClass.MEASUREMENT,
|
|
|
|
native_unit_of_measurement=ELECTRIC_CURRENT_AMPERE,
|
|
|
|
),
|
2022-01-31 22:48:16 +00:00
|
|
|
CharacteristicsTypes.VENDOR_KOOGEEK_REALTIME_ENERGY: HomeKitSensorEntityDescription(
|
|
|
|
key=CharacteristicsTypes.VENDOR_KOOGEEK_REALTIME_ENERGY,
|
2022-01-27 16:08:26 +00:00
|
|
|
name="Power",
|
2021-12-13 16:04:21 +00:00
|
|
|
device_class=SensorDeviceClass.POWER,
|
|
|
|
state_class=SensorStateClass.MEASUREMENT,
|
2021-10-03 23:42:25 +00:00
|
|
|
native_unit_of_measurement=POWER_WATT,
|
|
|
|
),
|
2022-01-31 22:48:16 +00:00
|
|
|
CharacteristicsTypes.VENDOR_KOOGEEK_REALTIME_ENERGY_2: HomeKitSensorEntityDescription(
|
|
|
|
key=CharacteristicsTypes.VENDOR_KOOGEEK_REALTIME_ENERGY_2,
|
2022-01-27 16:08:26 +00:00
|
|
|
name="Power",
|
2021-12-13 16:04:21 +00:00
|
|
|
device_class=SensorDeviceClass.POWER,
|
|
|
|
state_class=SensorStateClass.MEASUREMENT,
|
2021-10-03 23:42:25 +00:00
|
|
|
native_unit_of_measurement=POWER_WATT,
|
|
|
|
),
|
2022-01-31 22:48:16 +00:00
|
|
|
CharacteristicsTypes.VENDOR_EVE_DEGREE_AIR_PRESSURE: HomeKitSensorEntityDescription(
|
|
|
|
key=CharacteristicsTypes.VENDOR_EVE_DEGREE_AIR_PRESSURE,
|
2021-10-03 23:42:25 +00:00
|
|
|
name="Air Pressure",
|
2021-12-13 16:04:21 +00:00
|
|
|
device_class=SensorDeviceClass.PRESSURE,
|
|
|
|
state_class=SensorStateClass.MEASUREMENT,
|
2021-10-03 23:42:25 +00:00
|
|
|
native_unit_of_measurement=PRESSURE_HPA,
|
|
|
|
),
|
2022-01-31 22:48:16 +00:00
|
|
|
CharacteristicsTypes.VENDOR_VOCOLINC_OUTLET_ENERGY: HomeKitSensorEntityDescription(
|
|
|
|
key=CharacteristicsTypes.VENDOR_VOCOLINC_OUTLET_ENERGY,
|
2022-01-27 16:08:26 +00:00
|
|
|
name="Power",
|
2022-01-21 20:20:32 +00:00
|
|
|
device_class=SensorDeviceClass.POWER,
|
|
|
|
state_class=SensorStateClass.MEASUREMENT,
|
|
|
|
native_unit_of_measurement=POWER_WATT,
|
|
|
|
),
|
2021-10-03 23:42:25 +00:00
|
|
|
CharacteristicsTypes.TEMPERATURE_CURRENT: HomeKitSensorEntityDescription(
|
|
|
|
key=CharacteristicsTypes.TEMPERATURE_CURRENT,
|
|
|
|
name="Current Temperature",
|
2021-12-13 16:04:21 +00:00
|
|
|
device_class=SensorDeviceClass.TEMPERATURE,
|
|
|
|
state_class=SensorStateClass.MEASUREMENT,
|
2021-10-03 23:42:25 +00:00
|
|
|
native_unit_of_measurement=TEMP_CELSIUS,
|
2021-06-28 21:48:29 +00:00
|
|
|
# This sensor is only for temperature characteristics that are not part
|
|
|
|
# of a temperature sensor service.
|
2022-01-31 22:48:16 +00:00
|
|
|
probe=(lambda char: char.service.type != ServicesTypes.TEMPERATURE_SENSOR),
|
2021-10-03 23:42:25 +00:00
|
|
|
),
|
|
|
|
CharacteristicsTypes.RELATIVE_HUMIDITY_CURRENT: HomeKitSensorEntityDescription(
|
|
|
|
key=CharacteristicsTypes.RELATIVE_HUMIDITY_CURRENT,
|
|
|
|
name="Current Humidity",
|
2021-12-13 16:04:21 +00:00
|
|
|
device_class=SensorDeviceClass.HUMIDITY,
|
|
|
|
state_class=SensorStateClass.MEASUREMENT,
|
2021-10-03 23:42:25 +00:00
|
|
|
native_unit_of_measurement=PERCENTAGE,
|
2021-07-26 21:11:27 +00:00
|
|
|
# This sensor is only for humidity characteristics that are not part
|
|
|
|
# of a humidity sensor service.
|
2022-01-31 22:48:16 +00:00
|
|
|
probe=(lambda char: char.service.type != ServicesTypes.HUMIDITY_SENSOR),
|
2021-10-03 23:42:25 +00:00
|
|
|
),
|
|
|
|
CharacteristicsTypes.AIR_QUALITY: HomeKitSensorEntityDescription(
|
|
|
|
key=CharacteristicsTypes.AIR_QUALITY,
|
|
|
|
name="Air Quality",
|
2021-12-13 16:04:21 +00:00
|
|
|
device_class=SensorDeviceClass.AQI,
|
|
|
|
state_class=SensorStateClass.MEASUREMENT,
|
2021-10-03 23:42:25 +00:00
|
|
|
),
|
|
|
|
CharacteristicsTypes.DENSITY_PM25: HomeKitSensorEntityDescription(
|
|
|
|
key=CharacteristicsTypes.DENSITY_PM25,
|
|
|
|
name="PM2.5 Density",
|
2021-12-13 16:04:21 +00:00
|
|
|
device_class=SensorDeviceClass.PM25,
|
|
|
|
state_class=SensorStateClass.MEASUREMENT,
|
2021-10-03 23:42:25 +00:00
|
|
|
native_unit_of_measurement=CONCENTRATION_MICROGRAMS_PER_CUBIC_METER,
|
|
|
|
),
|
|
|
|
CharacteristicsTypes.DENSITY_PM10: HomeKitSensorEntityDescription(
|
|
|
|
key=CharacteristicsTypes.DENSITY_PM10,
|
|
|
|
name="PM10 Density",
|
2021-12-13 16:04:21 +00:00
|
|
|
device_class=SensorDeviceClass.PM10,
|
|
|
|
state_class=SensorStateClass.MEASUREMENT,
|
2021-10-03 23:42:25 +00:00
|
|
|
native_unit_of_measurement=CONCENTRATION_MICROGRAMS_PER_CUBIC_METER,
|
|
|
|
),
|
|
|
|
CharacteristicsTypes.DENSITY_OZONE: HomeKitSensorEntityDescription(
|
|
|
|
key=CharacteristicsTypes.DENSITY_OZONE,
|
|
|
|
name="Ozone Density",
|
2021-12-13 16:04:21 +00:00
|
|
|
device_class=SensorDeviceClass.OZONE,
|
|
|
|
state_class=SensorStateClass.MEASUREMENT,
|
2021-10-03 23:42:25 +00:00
|
|
|
native_unit_of_measurement=CONCENTRATION_MICROGRAMS_PER_CUBIC_METER,
|
|
|
|
),
|
|
|
|
CharacteristicsTypes.DENSITY_NO2: HomeKitSensorEntityDescription(
|
|
|
|
key=CharacteristicsTypes.DENSITY_NO2,
|
|
|
|
name="Nitrogen Dioxide Density",
|
2021-12-13 16:04:21 +00:00
|
|
|
device_class=SensorDeviceClass.NITROGEN_DIOXIDE,
|
|
|
|
state_class=SensorStateClass.MEASUREMENT,
|
2021-10-03 23:42:25 +00:00
|
|
|
native_unit_of_measurement=CONCENTRATION_MICROGRAMS_PER_CUBIC_METER,
|
|
|
|
),
|
|
|
|
CharacteristicsTypes.DENSITY_SO2: HomeKitSensorEntityDescription(
|
|
|
|
key=CharacteristicsTypes.DENSITY_SO2,
|
|
|
|
name="Sulphur Dioxide Density",
|
2021-12-13 16:04:21 +00:00
|
|
|
device_class=SensorDeviceClass.SULPHUR_DIOXIDE,
|
|
|
|
state_class=SensorStateClass.MEASUREMENT,
|
2021-10-03 23:42:25 +00:00
|
|
|
native_unit_of_measurement=CONCENTRATION_MICROGRAMS_PER_CUBIC_METER,
|
|
|
|
),
|
|
|
|
CharacteristicsTypes.DENSITY_VOC: HomeKitSensorEntityDescription(
|
|
|
|
key=CharacteristicsTypes.DENSITY_VOC,
|
|
|
|
name="Volatile Organic Compound Density",
|
2021-12-13 16:04:21 +00:00
|
|
|
device_class=SensorDeviceClass.VOLATILE_ORGANIC_COMPOUNDS,
|
|
|
|
state_class=SensorStateClass.MEASUREMENT,
|
2021-10-03 23:42:25 +00:00
|
|
|
native_unit_of_measurement=CONCENTRATION_MICROGRAMS_PER_CUBIC_METER,
|
|
|
|
),
|
2022-08-03 21:03:10 +00:00
|
|
|
CharacteristicsTypes.THREAD_NODE_CAPABILITIES: HomeKitSensorEntityDescription(
|
|
|
|
key=CharacteristicsTypes.THREAD_NODE_CAPABILITIES,
|
|
|
|
name="Thread Capabilities",
|
|
|
|
entity_category=EntityCategory.DIAGNOSTIC,
|
|
|
|
format=thread_node_capability_to_str,
|
2022-12-02 13:50:01 +00:00
|
|
|
device_class=SensorDeviceClass.ENUM,
|
|
|
|
options=[
|
|
|
|
"border_router_capable",
|
|
|
|
"full",
|
|
|
|
"minimal",
|
|
|
|
"none",
|
|
|
|
"router_eligible",
|
|
|
|
"sleepy",
|
|
|
|
],
|
|
|
|
translation_key="thread_node_capabilities",
|
2022-08-03 21:03:10 +00:00
|
|
|
),
|
2022-08-04 10:55:29 +00:00
|
|
|
CharacteristicsTypes.THREAD_STATUS: HomeKitSensorEntityDescription(
|
|
|
|
key=CharacteristicsTypes.THREAD_STATUS,
|
|
|
|
name="Thread Status",
|
|
|
|
entity_category=EntityCategory.DIAGNOSTIC,
|
|
|
|
format=thread_status_to_str,
|
2022-12-02 13:50:01 +00:00
|
|
|
device_class=SensorDeviceClass.ENUM,
|
|
|
|
options=[
|
|
|
|
"border_router",
|
|
|
|
"child",
|
|
|
|
"detached",
|
|
|
|
"disabled",
|
|
|
|
"joining",
|
|
|
|
"leader",
|
|
|
|
"router",
|
|
|
|
],
|
|
|
|
translation_key="thread_status",
|
2022-08-04 10:55:29 +00:00
|
|
|
),
|
2022-10-24 15:31:26 +00:00
|
|
|
CharacteristicsTypes.VENDOR_NETATMO_NOISE: HomeKitSensorEntityDescription(
|
|
|
|
key=CharacteristicsTypes.VENDOR_NETATMO_NOISE,
|
|
|
|
name="Noise",
|
|
|
|
state_class=SensorStateClass.MEASUREMENT,
|
|
|
|
native_unit_of_measurement=SOUND_PRESSURE_DB,
|
|
|
|
),
|
2021-01-26 19:45:01 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2022-07-23 16:14:06 +00:00
|
|
|
class HomeKitSensor(HomeKitEntity, SensorEntity):
|
2022-07-03 20:47:54 +00:00
|
|
|
"""Representation of a HomeKit sensor."""
|
|
|
|
|
2022-07-23 16:14:06 +00:00
|
|
|
_attr_state_class = SensorStateClass.MEASUREMENT
|
|
|
|
|
2022-07-03 20:47:54 +00:00
|
|
|
@property
|
|
|
|
def name(self) -> str | None:
|
|
|
|
"""Return the name of the device."""
|
|
|
|
full_name = super().name
|
|
|
|
default_name = self.default_name
|
|
|
|
if (
|
|
|
|
default_name
|
|
|
|
and full_name
|
|
|
|
and folded_name(default_name) not in folded_name(full_name)
|
|
|
|
):
|
|
|
|
return f"{full_name} {default_name}"
|
|
|
|
return full_name
|
|
|
|
|
|
|
|
|
2022-07-23 16:14:06 +00:00
|
|
|
class HomeKitHumiditySensor(HomeKitSensor):
|
2019-02-28 18:09:04 +00:00
|
|
|
"""Representation of a Homekit humidity sensor."""
|
|
|
|
|
2021-12-13 16:04:21 +00:00
|
|
|
_attr_device_class = SensorDeviceClass.HUMIDITY
|
2021-08-12 12:23:56 +00:00
|
|
|
_attr_native_unit_of_measurement = PERCENTAGE
|
2021-05-31 08:50:11 +00:00
|
|
|
|
2022-02-01 19:30:37 +00:00
|
|
|
def get_characteristic_types(self) -> list[str]:
|
2019-02-28 18:09:04 +00:00
|
|
|
"""Define the homekit characteristics the entity is tracking."""
|
2019-07-31 19:25:30 +00:00
|
|
|
return [CharacteristicsTypes.RELATIVE_HUMIDITY_CURRENT]
|
2019-02-28 18:09:04 +00:00
|
|
|
|
|
|
|
@property
|
2022-07-03 20:47:54 +00:00
|
|
|
def default_name(self) -> str:
|
|
|
|
"""Return the default name of the device."""
|
|
|
|
return "Humidity"
|
2019-02-28 18:09:04 +00:00
|
|
|
|
|
|
|
@property
|
2022-02-01 19:30:37 +00:00
|
|
|
def native_value(self) -> float:
|
2019-02-28 18:09:04 +00:00
|
|
|
"""Return the current humidity."""
|
2020-03-11 11:40:47 +00:00
|
|
|
return self.service.value(CharacteristicsTypes.RELATIVE_HUMIDITY_CURRENT)
|
2019-02-28 18:09:04 +00:00
|
|
|
|
|
|
|
|
2022-07-23 16:14:06 +00:00
|
|
|
class HomeKitTemperatureSensor(HomeKitSensor):
|
2019-02-28 18:09:04 +00:00
|
|
|
"""Representation of a Homekit temperature sensor."""
|
|
|
|
|
2021-12-13 16:04:21 +00:00
|
|
|
_attr_device_class = SensorDeviceClass.TEMPERATURE
|
2021-08-12 12:23:56 +00:00
|
|
|
_attr_native_unit_of_measurement = TEMP_CELSIUS
|
2021-05-31 08:50:11 +00:00
|
|
|
|
2022-02-01 19:30:37 +00:00
|
|
|
def get_characteristic_types(self) -> list[str]:
|
2019-02-28 18:09:04 +00:00
|
|
|
"""Define the homekit characteristics the entity is tracking."""
|
2019-07-31 19:25:30 +00:00
|
|
|
return [CharacteristicsTypes.TEMPERATURE_CURRENT]
|
2019-02-28 18:09:04 +00:00
|
|
|
|
|
|
|
@property
|
2022-07-03 20:47:54 +00:00
|
|
|
def default_name(self) -> str:
|
|
|
|
"""Return the default name of the device."""
|
|
|
|
return "Temperature"
|
2019-02-28 18:09:04 +00:00
|
|
|
|
|
|
|
@property
|
2022-02-01 19:30:37 +00:00
|
|
|
def native_value(self) -> float:
|
2019-02-28 18:09:04 +00:00
|
|
|
"""Return the current temperature in Celsius."""
|
2020-03-11 11:40:47 +00:00
|
|
|
return self.service.value(CharacteristicsTypes.TEMPERATURE_CURRENT)
|
2019-02-28 18:09:04 +00:00
|
|
|
|
|
|
|
|
2022-07-23 16:14:06 +00:00
|
|
|
class HomeKitLightSensor(HomeKitSensor):
|
2019-02-28 18:09:04 +00:00
|
|
|
"""Representation of a Homekit light level sensor."""
|
|
|
|
|
2021-12-13 16:04:21 +00:00
|
|
|
_attr_device_class = SensorDeviceClass.ILLUMINANCE
|
2021-08-12 12:23:56 +00:00
|
|
|
_attr_native_unit_of_measurement = LIGHT_LUX
|
2021-05-31 08:50:11 +00:00
|
|
|
|
2022-02-01 19:30:37 +00:00
|
|
|
def get_characteristic_types(self) -> list[str]:
|
2019-02-28 18:09:04 +00:00
|
|
|
"""Define the homekit characteristics the entity is tracking."""
|
2019-07-31 19:25:30 +00:00
|
|
|
return [CharacteristicsTypes.LIGHT_LEVEL_CURRENT]
|
2019-02-28 18:09:04 +00:00
|
|
|
|
|
|
|
@property
|
2022-07-03 20:47:54 +00:00
|
|
|
def default_name(self) -> str:
|
|
|
|
"""Return the default name of the device."""
|
|
|
|
return "Light Level"
|
2019-02-28 18:09:04 +00:00
|
|
|
|
|
|
|
@property
|
2022-02-01 19:30:37 +00:00
|
|
|
def native_value(self) -> int:
|
2019-02-28 18:09:04 +00:00
|
|
|
"""Return the current light level in lux."""
|
2020-03-11 11:40:47 +00:00
|
|
|
return self.service.value(CharacteristicsTypes.LIGHT_LEVEL_CURRENT)
|
2019-08-01 10:24:46 +00:00
|
|
|
|
|
|
|
|
2022-07-23 16:14:06 +00:00
|
|
|
class HomeKitCarbonDioxideSensor(HomeKitSensor):
|
2019-08-01 10:24:46 +00:00
|
|
|
"""Representation of a Homekit Carbon Dioxide sensor."""
|
|
|
|
|
2022-07-23 16:14:06 +00:00
|
|
|
_attr_device_class = SensorDeviceClass.CO2
|
2021-08-12 12:23:56 +00:00
|
|
|
_attr_native_unit_of_measurement = CONCENTRATION_PARTS_PER_MILLION
|
2021-05-31 08:50:11 +00:00
|
|
|
|
2022-02-01 19:30:37 +00:00
|
|
|
def get_characteristic_types(self) -> list[str]:
|
2019-08-01 10:24:46 +00:00
|
|
|
"""Define the homekit characteristics the entity is tracking."""
|
|
|
|
return [CharacteristicsTypes.CARBON_DIOXIDE_LEVEL]
|
|
|
|
|
|
|
|
@property
|
2022-07-03 20:47:54 +00:00
|
|
|
def default_name(self) -> str:
|
|
|
|
"""Return the default name of the device."""
|
2022-07-23 16:14:06 +00:00
|
|
|
return "Carbon Dioxide"
|
2019-08-01 10:24:46 +00:00
|
|
|
|
|
|
|
@property
|
2022-02-01 19:30:37 +00:00
|
|
|
def native_value(self) -> int:
|
2019-08-01 10:24:46 +00:00
|
|
|
"""Return the current CO2 level in ppm."""
|
2020-03-11 11:40:47 +00:00
|
|
|
return self.service.value(CharacteristicsTypes.CARBON_DIOXIDE_LEVEL)
|
2019-08-01 10:24:46 +00:00
|
|
|
|
|
|
|
|
2022-07-23 16:14:06 +00:00
|
|
|
class HomeKitBatterySensor(HomeKitSensor):
|
2019-08-31 12:18:18 +00:00
|
|
|
"""Representation of a Homekit battery sensor."""
|
|
|
|
|
2021-12-13 16:04:21 +00:00
|
|
|
_attr_device_class = SensorDeviceClass.BATTERY
|
2021-08-12 12:23:56 +00:00
|
|
|
_attr_native_unit_of_measurement = PERCENTAGE
|
2022-09-25 22:08:28 +00:00
|
|
|
_attr_entity_category = EntityCategory.DIAGNOSTIC
|
2021-05-31 08:50:11 +00:00
|
|
|
|
2022-02-01 19:30:37 +00:00
|
|
|
def get_characteristic_types(self) -> list[str]:
|
2019-08-31 12:18:18 +00:00
|
|
|
"""Define the homekit characteristics the entity is tracking."""
|
|
|
|
return [
|
|
|
|
CharacteristicsTypes.BATTERY_LEVEL,
|
|
|
|
CharacteristicsTypes.STATUS_LO_BATT,
|
|
|
|
CharacteristicsTypes.CHARGING_STATE,
|
|
|
|
]
|
|
|
|
|
|
|
|
@property
|
2022-07-03 20:47:54 +00:00
|
|
|
def default_name(self) -> str:
|
|
|
|
"""Return the default name of the device."""
|
|
|
|
return "Battery"
|
2019-08-31 12:18:18 +00:00
|
|
|
|
|
|
|
@property
|
2022-02-01 19:30:37 +00:00
|
|
|
def icon(self) -> str:
|
2019-08-31 12:18:18 +00:00
|
|
|
"""Return the sensor icon."""
|
|
|
|
if not self.available or self.state is None:
|
|
|
|
return "mdi:battery-unknown"
|
|
|
|
|
|
|
|
# This is similar to the logic in helpers.icon, but we have delegated the
|
|
|
|
# decision about what mdi:battery-alert is to the device.
|
|
|
|
icon = "mdi:battery"
|
2020-03-11 11:40:47 +00:00
|
|
|
if self.is_charging and self.state > 10:
|
2019-08-31 12:18:18 +00:00
|
|
|
percentage = int(round(self.state / 20 - 0.01)) * 20
|
|
|
|
icon += f"-charging-{percentage}"
|
2020-03-11 11:40:47 +00:00
|
|
|
elif self.is_charging:
|
2019-08-31 12:18:18 +00:00
|
|
|
icon += "-outline"
|
2020-03-11 11:40:47 +00:00
|
|
|
elif self.is_low_battery:
|
2019-08-31 12:18:18 +00:00
|
|
|
icon += "-alert"
|
|
|
|
elif self.state < 95:
|
|
|
|
percentage = max(int(round(self.state / 10 - 0.01)) * 10, 10)
|
|
|
|
icon += f"-{percentage}"
|
|
|
|
|
|
|
|
return icon
|
|
|
|
|
2020-03-11 11:40:47 +00:00
|
|
|
@property
|
2022-02-01 19:30:37 +00:00
|
|
|
def is_low_battery(self) -> bool:
|
2020-03-11 11:40:47 +00:00
|
|
|
"""Return true if battery level is low."""
|
|
|
|
return self.service.value(CharacteristicsTypes.STATUS_LO_BATT) == 1
|
2019-08-31 12:18:18 +00:00
|
|
|
|
2020-03-11 11:40:47 +00:00
|
|
|
@property
|
2022-02-01 19:30:37 +00:00
|
|
|
def is_charging(self) -> bool:
|
2020-03-11 11:40:47 +00:00
|
|
|
"""Return true if currently charing."""
|
2019-08-31 12:18:18 +00:00
|
|
|
# 0 = not charging
|
|
|
|
# 1 = charging
|
|
|
|
# 2 = not chargeable
|
2020-03-11 11:40:47 +00:00
|
|
|
return self.service.value(CharacteristicsTypes.CHARGING_STATE) == 1
|
2019-08-31 12:18:18 +00:00
|
|
|
|
|
|
|
@property
|
2022-02-01 19:30:37 +00:00
|
|
|
def native_value(self) -> int:
|
2019-08-31 12:18:18 +00:00
|
|
|
"""Return the current battery level percentage."""
|
2020-03-11 11:40:47 +00:00
|
|
|
return self.service.value(CharacteristicsTypes.BATTERY_LEVEL)
|
2019-08-31 12:18:18 +00:00
|
|
|
|
|
|
|
|
2021-03-22 18:45:17 +00:00
|
|
|
class SimpleSensor(CharacteristicEntity, SensorEntity):
|
2021-01-26 19:45:01 +00:00
|
|
|
"""
|
|
|
|
A simple sensor for a single characteristic.
|
|
|
|
|
|
|
|
This may be an additional secondary entity that is part of another service. An
|
|
|
|
example is a switch that has an energy sensor.
|
|
|
|
|
|
|
|
These *have* to have a different unique_id to the normal sensors as there could
|
|
|
|
be multiple entities per HomeKit service (this was not previously the case).
|
|
|
|
"""
|
|
|
|
|
2021-10-03 23:42:25 +00:00
|
|
|
entity_description: HomeKitSensorEntityDescription
|
|
|
|
|
2021-01-26 19:45:01 +00:00
|
|
|
def __init__(
|
|
|
|
self,
|
2022-02-01 19:30:37 +00:00
|
|
|
conn: HKDevice,
|
|
|
|
info: ConfigType,
|
|
|
|
char: Characteristic,
|
2021-10-03 23:42:25 +00:00
|
|
|
description: HomeKitSensorEntityDescription,
|
2022-02-01 19:30:37 +00:00
|
|
|
) -> None:
|
2021-01-26 19:45:01 +00:00
|
|
|
"""Initialise a secondary HomeKit characteristic sensor."""
|
2021-10-03 23:42:25 +00:00
|
|
|
self.entity_description = description
|
2021-08-02 03:59:32 +00:00
|
|
|
super().__init__(conn, info, char)
|
2021-01-26 19:45:01 +00:00
|
|
|
|
2022-02-01 19:30:37 +00:00
|
|
|
def get_characteristic_types(self) -> list[str]:
|
2021-01-26 19:45:01 +00:00
|
|
|
"""Define the homekit characteristics the entity is tracking."""
|
|
|
|
return [self._char.type]
|
|
|
|
|
|
|
|
@property
|
|
|
|
def name(self) -> str:
|
|
|
|
"""Return the name of the device if any."""
|
2022-07-03 20:47:54 +00:00
|
|
|
if name := self.accessory.name:
|
|
|
|
return f"{name} {self.entity_description.name}"
|
|
|
|
return f"{self.entity_description.name}"
|
2021-01-26 19:45:01 +00:00
|
|
|
|
|
|
|
@property
|
2022-02-01 19:30:37 +00:00
|
|
|
def native_value(self) -> str | int | float:
|
2021-01-26 19:45:01 +00:00
|
|
|
"""Return the current sensor value."""
|
2022-08-03 21:03:10 +00:00
|
|
|
val = self._char.value
|
|
|
|
if self.entity_description.format:
|
|
|
|
return self.entity_description.format(val)
|
|
|
|
return val
|
2021-01-26 19:45:01 +00:00
|
|
|
|
|
|
|
|
2019-08-01 10:24:46 +00:00
|
|
|
ENTITY_TYPES = {
|
2020-11-16 23:11:39 +00:00
|
|
|
ServicesTypes.HUMIDITY_SENSOR: HomeKitHumiditySensor,
|
|
|
|
ServicesTypes.TEMPERATURE_SENSOR: HomeKitTemperatureSensor,
|
|
|
|
ServicesTypes.LIGHT_SENSOR: HomeKitLightSensor,
|
|
|
|
ServicesTypes.CARBON_DIOXIDE_SENSOR: HomeKitCarbonDioxideSensor,
|
|
|
|
ServicesTypes.BATTERY_SERVICE: HomeKitBatterySensor,
|
2019-08-01 10:24:46 +00:00
|
|
|
}
|
|
|
|
|
2022-09-25 22:08:28 +00:00
|
|
|
# Only create the entity if it has the required characteristic
|
|
|
|
REQUIRED_CHAR_BY_TYPE = {
|
|
|
|
ServicesTypes.BATTERY_SERVICE: CharacteristicsTypes.BATTERY_LEVEL,
|
|
|
|
}
|
|
|
|
|
2019-08-01 10:24:46 +00:00
|
|
|
|
2022-09-26 01:31:56 +00:00
|
|
|
class RSSISensor(HomeKitEntity, SensorEntity):
|
|
|
|
"""HomeKit Controller RSSI sensor."""
|
|
|
|
|
|
|
|
_attr_device_class = SensorDeviceClass.SIGNAL_STRENGTH
|
|
|
|
_attr_entity_category = EntityCategory.DIAGNOSTIC
|
|
|
|
_attr_entity_registry_enabled_default = False
|
|
|
|
_attr_has_entity_name = True
|
|
|
|
_attr_native_unit_of_measurement = SIGNAL_STRENGTH_DECIBELS_MILLIWATT
|
|
|
|
_attr_should_poll = False
|
|
|
|
|
|
|
|
def get_characteristic_types(self) -> list[str]:
|
|
|
|
"""Define the homekit characteristics the entity cares about."""
|
|
|
|
return []
|
|
|
|
|
|
|
|
@property
|
|
|
|
def available(self) -> bool:
|
|
|
|
"""Return if the bluetooth device is available."""
|
|
|
|
address = self._accessory.pairing_data["AccessoryAddress"]
|
|
|
|
return async_ble_device_from_address(self.hass, address) is not None
|
|
|
|
|
|
|
|
@property
|
|
|
|
def name(self) -> str:
|
|
|
|
"""Return the name of the sensor."""
|
|
|
|
return "Signal strength"
|
|
|
|
|
|
|
|
@property
|
2022-10-11 21:26:03 +00:00
|
|
|
def old_unique_id(self) -> str:
|
|
|
|
"""Return the old ID of this device."""
|
2022-09-26 01:31:56 +00:00
|
|
|
serial = self.accessory_info.value(CharacteristicsTypes.SERIAL_NUMBER)
|
|
|
|
return f"homekit-{serial}-rssi"
|
|
|
|
|
2022-10-11 21:26:03 +00:00
|
|
|
@property
|
|
|
|
def unique_id(self) -> str:
|
|
|
|
"""Return the ID of this device."""
|
|
|
|
return f"{self._accessory.unique_id}_rssi"
|
|
|
|
|
2022-09-26 01:31:56 +00:00
|
|
|
@property
|
|
|
|
def native_value(self) -> int | None:
|
|
|
|
"""Return the current rssi value."""
|
|
|
|
address = self._accessory.pairing_data["AccessoryAddress"]
|
2022-10-15 17:57:23 +00:00
|
|
|
last_service_info = async_last_service_info(self.hass, address)
|
|
|
|
return last_service_info.rssi if last_service_info else None
|
2022-09-26 01:31:56 +00:00
|
|
|
|
|
|
|
|
2021-12-28 18:24:40 +00:00
|
|
|
async def async_setup_entry(
|
|
|
|
hass: HomeAssistant,
|
|
|
|
config_entry: ConfigEntry,
|
|
|
|
async_add_entities: AddEntitiesCallback,
|
|
|
|
) -> None:
|
2019-08-01 10:24:46 +00:00
|
|
|
"""Set up Homekit sensors."""
|
|
|
|
hkid = config_entry.data["AccessoryPairingID"]
|
2022-09-26 01:31:56 +00:00
|
|
|
conn: HKDevice = hass.data[KNOWN_DEVICES][hkid]
|
2019-08-01 10:24:46 +00:00
|
|
|
|
2020-01-29 21:59:45 +00:00
|
|
|
@callback
|
2022-02-01 19:30:37 +00:00
|
|
|
def async_add_service(service: Service) -> bool:
|
2022-01-31 22:48:16 +00:00
|
|
|
if not (entity_class := ENTITY_TYPES.get(service.type)):
|
2019-08-01 10:24:46 +00:00
|
|
|
return False
|
2022-09-25 22:08:28 +00:00
|
|
|
if (
|
|
|
|
required_char := REQUIRED_CHAR_BY_TYPE.get(service.type)
|
|
|
|
) and not service.has(required_char):
|
|
|
|
return False
|
2020-11-16 23:11:39 +00:00
|
|
|
info = {"aid": service.accessory.aid, "iid": service.iid}
|
2022-10-11 21:26:03 +00:00
|
|
|
entity: HomeKitSensor = entity_class(conn, info)
|
|
|
|
conn.async_migrate_unique_id(
|
|
|
|
entity.old_unique_id, entity.unique_id, Platform.SENSOR
|
|
|
|
)
|
|
|
|
async_add_entities([entity])
|
2019-08-01 10:24:46 +00:00
|
|
|
return True
|
|
|
|
|
|
|
|
conn.add_listener(async_add_service)
|
2021-01-26 19:45:01 +00:00
|
|
|
|
|
|
|
@callback
|
2022-02-01 19:30:37 +00:00
|
|
|
def async_add_characteristic(char: Characteristic) -> bool:
|
2021-10-03 23:42:25 +00:00
|
|
|
if not (description := SIMPLE_SENSOR.get(char.type)):
|
2021-01-26 19:45:01 +00:00
|
|
|
return False
|
2021-10-03 23:42:25 +00:00
|
|
|
if description.probe and not description.probe(char):
|
2021-06-28 21:48:29 +00:00
|
|
|
return False
|
2021-01-26 19:45:01 +00:00
|
|
|
info = {"aid": char.service.accessory.aid, "iid": char.service.iid}
|
2022-10-11 21:26:03 +00:00
|
|
|
entity = SimpleSensor(conn, info, char, description)
|
|
|
|
conn.async_migrate_unique_id(
|
|
|
|
entity.old_unique_id, entity.unique_id, Platform.SENSOR
|
|
|
|
)
|
|
|
|
async_add_entities([entity])
|
2021-01-26 19:45:01 +00:00
|
|
|
|
|
|
|
return True
|
|
|
|
|
|
|
|
conn.add_char_factory(async_add_characteristic)
|
2022-09-26 01:31:56 +00:00
|
|
|
|
|
|
|
@callback
|
|
|
|
def async_add_accessory(accessory: Accessory) -> bool:
|
|
|
|
if conn.pairing.transport != Transport.BLE:
|
|
|
|
return False
|
|
|
|
|
|
|
|
accessory_info = accessory.services.first(
|
|
|
|
service_type=ServicesTypes.ACCESSORY_INFORMATION
|
|
|
|
)
|
|
|
|
info = {"aid": accessory.aid, "iid": accessory_info.iid}
|
2022-10-11 21:26:03 +00:00
|
|
|
entity = RSSISensor(conn, info)
|
|
|
|
conn.async_migrate_unique_id(
|
|
|
|
entity.old_unique_id, entity.unique_id, Platform.SENSOR
|
|
|
|
)
|
|
|
|
async_add_entities([entity])
|
2022-09-26 01:31:56 +00:00
|
|
|
return True
|
|
|
|
|
|
|
|
conn.add_accessory_factory(async_add_accessory)
|