234 lines
7.1 KiB
Python
234 lines
7.1 KiB
Python
"""Support for Nest Thermostat sensors for the legacy API."""
|
|
# mypy: ignore-errors
|
|
|
|
import logging
|
|
|
|
from homeassistant.components.sensor import (
|
|
SensorDeviceClass,
|
|
SensorEntity,
|
|
SensorStateClass,
|
|
)
|
|
from homeassistant.config_entries import ConfigEntry
|
|
from homeassistant.const import (
|
|
CONF_MONITORED_CONDITIONS,
|
|
CONF_SENSORS,
|
|
PERCENTAGE,
|
|
STATE_OFF,
|
|
UnitOfTemperature,
|
|
)
|
|
from homeassistant.core import HomeAssistant
|
|
from homeassistant.helpers.entity_platform import AddEntitiesCallback
|
|
|
|
from . import NestSensorDevice
|
|
from .const import DATA_NEST, DATA_NEST_CONFIG
|
|
|
|
SENSOR_TYPES = ["humidity", "operation_mode", "hvac_state"]
|
|
|
|
TEMP_SENSOR_TYPES = ["temperature", "target"]
|
|
|
|
PROTECT_SENSOR_TYPES = [
|
|
"co_status",
|
|
"smoke_status",
|
|
"battery_health",
|
|
# color_status: "gray", "green", "yellow", "red"
|
|
"color_status",
|
|
]
|
|
|
|
STRUCTURE_SENSOR_TYPES = ["eta"]
|
|
|
|
STATE_HEAT = "heat"
|
|
STATE_COOL = "cool"
|
|
|
|
# security_state is structure level sensor, but only meaningful when
|
|
# Nest Cam exist
|
|
STRUCTURE_CAMERA_SENSOR_TYPES = ["security_state"]
|
|
|
|
_VALID_SENSOR_TYPES = (
|
|
SENSOR_TYPES
|
|
+ TEMP_SENSOR_TYPES
|
|
+ PROTECT_SENSOR_TYPES
|
|
+ STRUCTURE_SENSOR_TYPES
|
|
+ STRUCTURE_CAMERA_SENSOR_TYPES
|
|
)
|
|
|
|
SENSOR_UNITS = {"humidity": PERCENTAGE}
|
|
|
|
SENSOR_DEVICE_CLASSES = {"humidity": SensorDeviceClass.HUMIDITY}
|
|
|
|
SENSOR_STATE_CLASSES = {"humidity": SensorStateClass.MEASUREMENT}
|
|
|
|
VARIABLE_NAME_MAPPING = {"eta": "eta_begin", "operation_mode": "mode"}
|
|
|
|
VALUE_MAPPING = {
|
|
"hvac_state": {"heating": STATE_HEAT, "cooling": STATE_COOL, "off": STATE_OFF}
|
|
}
|
|
|
|
SENSOR_TYPES_DEPRECATED = ["last_ip", "local_ip", "last_connection", "battery_level"]
|
|
|
|
DEPRECATED_WEATHER_VARS = [
|
|
"weather_humidity",
|
|
"weather_temperature",
|
|
"weather_condition",
|
|
"wind_speed",
|
|
"wind_direction",
|
|
]
|
|
|
|
_SENSOR_TYPES_DEPRECATED = SENSOR_TYPES_DEPRECATED + DEPRECATED_WEATHER_VARS
|
|
|
|
_LOGGER = logging.getLogger(__name__)
|
|
|
|
|
|
async def async_setup_legacy_entry(
|
|
hass: HomeAssistant, entry: ConfigEntry, async_add_entities: AddEntitiesCallback
|
|
) -> None:
|
|
"""Set up a Nest sensor based on a config entry."""
|
|
nest = hass.data[DATA_NEST]
|
|
|
|
discovery_info = hass.data.get(DATA_NEST_CONFIG, {}).get(CONF_SENSORS, {})
|
|
|
|
# Add all available sensors if no Nest sensor config is set
|
|
if discovery_info == {}:
|
|
conditions = _VALID_SENSOR_TYPES
|
|
else:
|
|
conditions = discovery_info.get(CONF_MONITORED_CONDITIONS, {})
|
|
|
|
for variable in conditions:
|
|
if variable in _SENSOR_TYPES_DEPRECATED:
|
|
if variable in DEPRECATED_WEATHER_VARS:
|
|
wstr = (
|
|
f"Nest no longer provides weather data like {variable}. See "
|
|
"https://www.home-assistant.io/integrations/#weather "
|
|
"for a list of other weather integrations to use."
|
|
)
|
|
else:
|
|
wstr = (
|
|
f"{variable} is no a longer supported "
|
|
"monitored_conditions. See "
|
|
"https://www.home-assistant.io/integrations/"
|
|
"binary_sensor.nest/ for valid options."
|
|
)
|
|
_LOGGER.error(wstr)
|
|
|
|
def get_sensors():
|
|
"""Get the Nest sensors."""
|
|
all_sensors = []
|
|
for structure in nest.structures():
|
|
all_sensors += [
|
|
NestBasicSensor(structure, None, variable)
|
|
for variable in conditions
|
|
if variable in STRUCTURE_SENSOR_TYPES
|
|
]
|
|
|
|
for structure, device in nest.thermostats():
|
|
all_sensors += [
|
|
NestBasicSensor(structure, device, variable)
|
|
for variable in conditions
|
|
if variable in SENSOR_TYPES
|
|
]
|
|
all_sensors += [
|
|
NestTempSensor(structure, device, variable)
|
|
for variable in conditions
|
|
if variable in TEMP_SENSOR_TYPES
|
|
]
|
|
|
|
for structure, device in nest.smoke_co_alarms():
|
|
all_sensors += [
|
|
NestBasicSensor(structure, device, variable)
|
|
for variable in conditions
|
|
if variable in PROTECT_SENSOR_TYPES
|
|
]
|
|
|
|
structures_has_camera = {}
|
|
for structure, _ in nest.cameras():
|
|
structures_has_camera[structure] = True
|
|
for structure in structures_has_camera:
|
|
all_sensors += [
|
|
NestBasicSensor(structure, None, variable)
|
|
for variable in conditions
|
|
if variable in STRUCTURE_CAMERA_SENSOR_TYPES
|
|
]
|
|
|
|
return all_sensors
|
|
|
|
async_add_entities(await hass.async_add_executor_job(get_sensors), True)
|
|
|
|
|
|
class NestBasicSensor(NestSensorDevice, SensorEntity):
|
|
"""Representation a basic Nest sensor."""
|
|
|
|
@property
|
|
def native_unit_of_measurement(self):
|
|
"""Return the unit the value is expressed in."""
|
|
return self._unit
|
|
|
|
@property
|
|
def native_value(self):
|
|
"""Return the state of the sensor."""
|
|
return self._state
|
|
|
|
@property
|
|
def device_class(self):
|
|
"""Return the device class of the sensor."""
|
|
return SENSOR_DEVICE_CLASSES.get(self.variable)
|
|
|
|
@property
|
|
def state_class(self):
|
|
"""Return the state class of the sensor."""
|
|
return SENSOR_STATE_CLASSES.get(self.variable)
|
|
|
|
def update(self):
|
|
"""Retrieve latest state."""
|
|
self._unit = SENSOR_UNITS.get(self.variable)
|
|
|
|
if self.variable in VARIABLE_NAME_MAPPING:
|
|
self._state = getattr(self.device, VARIABLE_NAME_MAPPING[self.variable])
|
|
elif self.variable in VALUE_MAPPING:
|
|
state = getattr(self.device, self.variable)
|
|
self._state = VALUE_MAPPING[self.variable].get(state, state)
|
|
elif self.variable in PROTECT_SENSOR_TYPES and self.variable != "color_status":
|
|
# keep backward compatibility
|
|
state = getattr(self.device, self.variable)
|
|
self._state = state.capitalize() if state is not None else None
|
|
else:
|
|
self._state = getattr(self.device, self.variable)
|
|
|
|
|
|
class NestTempSensor(NestSensorDevice, SensorEntity):
|
|
"""Representation of a Nest Temperature sensor."""
|
|
|
|
@property
|
|
def native_value(self):
|
|
"""Return the state of the sensor."""
|
|
return self._state
|
|
|
|
@property
|
|
def native_unit_of_measurement(self):
|
|
"""Return the unit the value is expressed in."""
|
|
return self._unit
|
|
|
|
@property
|
|
def device_class(self):
|
|
"""Return the device class of the sensor."""
|
|
return SensorDeviceClass.TEMPERATURE
|
|
|
|
@property
|
|
def state_class(self):
|
|
"""Return the state class of the sensor."""
|
|
return SensorStateClass.MEASUREMENT
|
|
|
|
def update(self):
|
|
"""Retrieve latest state."""
|
|
if self.device.temperature_scale == "C":
|
|
self._unit = UnitOfTemperature.CELSIUS
|
|
else:
|
|
self._unit = UnitOfTemperature.FAHRENHEIT
|
|
|
|
if (temp := getattr(self.device, self.variable)) is None:
|
|
self._state = None
|
|
|
|
if isinstance(temp, tuple):
|
|
low, high = temp
|
|
self._state = f"{int(low)}-{int(high)}"
|
|
else:
|
|
self._state = round(temp, 1)
|