mirror of
https://github.com/home-assistant/core.git
synced 2025-07-17 18:27:09 +00:00
Support Nobø Switch as temperature sensor (#78480)
* Expose Nobø Switch as temperatur sensor. * - Review - Hub may report current temperature as None * Avoid update during entity addition, and fix race condition * Update pynobo to 1.6.0 Use new method to fix potential race condition. * Use generator expressions
This commit is contained in:
parent
073951177b
commit
712b984833
@ -860,6 +860,7 @@ omit =
|
|||||||
homeassistant/components/noaa_tides/sensor.py
|
homeassistant/components/noaa_tides/sensor.py
|
||||||
homeassistant/components/nobo_hub/__init__.py
|
homeassistant/components/nobo_hub/__init__.py
|
||||||
homeassistant/components/nobo_hub/climate.py
|
homeassistant/components/nobo_hub/climate.py
|
||||||
|
homeassistant/components/nobo_hub/sensor.py
|
||||||
homeassistant/components/norway_air/air_quality.py
|
homeassistant/components/norway_air/air_quality.py
|
||||||
homeassistant/components/notify_events/notify.py
|
homeassistant/components/notify_events/notify.py
|
||||||
homeassistant/components/notion/__init__.py
|
homeassistant/components/notion/__init__.py
|
||||||
|
@ -1,8 +1,6 @@
|
|||||||
"""The Nobø Ecohub integration."""
|
"""The Nobø Ecohub integration."""
|
||||||
from __future__ import annotations
|
from __future__ import annotations
|
||||||
|
|
||||||
import logging
|
|
||||||
|
|
||||||
from pynobo import nobo
|
from pynobo import nobo
|
||||||
|
|
||||||
from homeassistant.config_entries import ConfigEntry
|
from homeassistant.config_entries import ConfigEntry
|
||||||
@ -25,9 +23,7 @@ from .const import (
|
|||||||
NOBO_MANUFACTURER,
|
NOBO_MANUFACTURER,
|
||||||
)
|
)
|
||||||
|
|
||||||
PLATFORMS = [Platform.CLIMATE]
|
PLATFORMS = [Platform.CLIMATE, Platform.SENSOR]
|
||||||
|
|
||||||
_LOGGER = logging.getLogger(__name__)
|
|
||||||
|
|
||||||
|
|
||||||
async def async_setup_entry(hass: HomeAssistant, entry: ConfigEntry) -> bool:
|
async def async_setup_entry(hass: HomeAssistant, entry: ConfigEntry) -> bool:
|
||||||
@ -37,7 +33,7 @@ async def async_setup_entry(hass: HomeAssistant, entry: ConfigEntry) -> bool:
|
|||||||
discover = entry.data[CONF_AUTO_DISCOVERED]
|
discover = entry.data[CONF_AUTO_DISCOVERED]
|
||||||
ip_address = None if discover else entry.data[CONF_IP_ADDRESS]
|
ip_address = None if discover else entry.data[CONF_IP_ADDRESS]
|
||||||
hub = nobo(serial=serial, ip=ip_address, discover=discover, synchronous=False)
|
hub = nobo(serial=serial, ip=ip_address, discover=discover, synchronous=False)
|
||||||
await hub.start()
|
await hub.connect()
|
||||||
|
|
||||||
hass.data.setdefault(DOMAIN, {})
|
hass.data.setdefault(DOMAIN, {})
|
||||||
|
|
||||||
@ -65,6 +61,8 @@ async def async_setup_entry(hass: HomeAssistant, entry: ConfigEntry) -> bool:
|
|||||||
|
|
||||||
entry.async_on_unload(entry.add_update_listener(options_update_listener))
|
entry.async_on_unload(entry.add_update_listener(options_update_listener))
|
||||||
|
|
||||||
|
await hub.start()
|
||||||
|
|
||||||
return True
|
return True
|
||||||
|
|
||||||
|
|
||||||
|
@ -69,10 +69,7 @@ async def async_setup_entry(
|
|||||||
)
|
)
|
||||||
|
|
||||||
# Add zones as entities
|
# Add zones as entities
|
||||||
async_add_entities(
|
async_add_entities(NoboZone(zone_id, hub, override_type) for zone_id in hub.zones)
|
||||||
[NoboZone(zone_id, hub, override_type) for zone_id in hub.zones],
|
|
||||||
True,
|
|
||||||
)
|
|
||||||
|
|
||||||
|
|
||||||
class NoboZone(ClimateEntity):
|
class NoboZone(ClimateEntity):
|
||||||
@ -107,6 +104,7 @@ class NoboZone(ClimateEntity):
|
|||||||
ATTR_VIA_DEVICE: (DOMAIN, hub.hub_info[ATTR_SERIAL]),
|
ATTR_VIA_DEVICE: (DOMAIN, hub.hub_info[ATTR_SERIAL]),
|
||||||
ATTR_SUGGESTED_AREA: hub.zones[zone_id][ATTR_NAME],
|
ATTR_SUGGESTED_AREA: hub.zones[zone_id][ATTR_NAME],
|
||||||
}
|
}
|
||||||
|
self._read_state()
|
||||||
|
|
||||||
async def async_added_to_hass(self) -> None:
|
async def async_added_to_hass(self) -> None:
|
||||||
"""Register callback from hub."""
|
"""Register callback from hub."""
|
||||||
|
@ -17,3 +17,4 @@ ATTR_TEMP_ECO_C = "temp_eco_c"
|
|||||||
ATTR_OVERRIDE_ALLOWED = "override_allowed"
|
ATTR_OVERRIDE_ALLOWED = "override_allowed"
|
||||||
ATTR_TARGET_TYPE = "target_type"
|
ATTR_TARGET_TYPE = "target_type"
|
||||||
ATTR_TARGET_ID = "target_id"
|
ATTR_TARGET_ID = "target_id"
|
||||||
|
ATTR_ZONE_ID = "zone_id"
|
||||||
|
@ -3,7 +3,7 @@
|
|||||||
"name": "Nob\u00f8 Ecohub",
|
"name": "Nob\u00f8 Ecohub",
|
||||||
"config_flow": true,
|
"config_flow": true,
|
||||||
"documentation": "https://www.home-assistant.io/integrations/nobo_hub",
|
"documentation": "https://www.home-assistant.io/integrations/nobo_hub",
|
||||||
"requirements": ["pynobo==1.4.0"],
|
"requirements": ["pynobo==1.6.0"],
|
||||||
"codeowners": ["@echoromeo", "@oyvindwe"],
|
"codeowners": ["@echoromeo", "@oyvindwe"],
|
||||||
"iot_class": "local_push"
|
"iot_class": "local_push"
|
||||||
}
|
}
|
||||||
|
95
homeassistant/components/nobo_hub/sensor.py
Normal file
95
homeassistant/components/nobo_hub/sensor.py
Normal file
@ -0,0 +1,95 @@
|
|||||||
|
"""Python Control of Nobø Hub - Nobø Energy Control."""
|
||||||
|
from __future__ import annotations
|
||||||
|
|
||||||
|
from pynobo import nobo
|
||||||
|
|
||||||
|
from homeassistant.components.sensor import (
|
||||||
|
SensorDeviceClass,
|
||||||
|
SensorEntity,
|
||||||
|
SensorStateClass,
|
||||||
|
)
|
||||||
|
from homeassistant.config_entries import ConfigEntry
|
||||||
|
from homeassistant.const import (
|
||||||
|
ATTR_IDENTIFIERS,
|
||||||
|
ATTR_MANUFACTURER,
|
||||||
|
ATTR_MODEL,
|
||||||
|
ATTR_NAME,
|
||||||
|
ATTR_SUGGESTED_AREA,
|
||||||
|
ATTR_VIA_DEVICE,
|
||||||
|
TEMP_CELSIUS,
|
||||||
|
)
|
||||||
|
from homeassistant.core import HomeAssistant, callback
|
||||||
|
from homeassistant.helpers.entity import DeviceInfo
|
||||||
|
from homeassistant.helpers.entity_platform import AddEntitiesCallback
|
||||||
|
from homeassistant.helpers.typing import StateType
|
||||||
|
|
||||||
|
from .const import ATTR_SERIAL, ATTR_ZONE_ID, DOMAIN, NOBO_MANUFACTURER
|
||||||
|
|
||||||
|
|
||||||
|
async def async_setup_entry(
|
||||||
|
hass: HomeAssistant,
|
||||||
|
config_entry: ConfigEntry,
|
||||||
|
async_add_entities: AddEntitiesCallback,
|
||||||
|
) -> None:
|
||||||
|
"""Set up any temperature sensors connected to the Nobø Ecohub."""
|
||||||
|
|
||||||
|
# Setup connection with hub
|
||||||
|
hub: nobo = hass.data[DOMAIN][config_entry.entry_id]
|
||||||
|
|
||||||
|
async_add_entities(
|
||||||
|
NoboTemperatureSensor(component["serial"], hub)
|
||||||
|
for component in hub.components.values()
|
||||||
|
if component[ATTR_MODEL].has_temp_sensor
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
|
class NoboTemperatureSensor(SensorEntity):
|
||||||
|
"""A Nobø device with a temperature sensor."""
|
||||||
|
|
||||||
|
_attr_device_class = SensorDeviceClass.TEMPERATURE
|
||||||
|
_attr_native_unit_of_measurement = TEMP_CELSIUS
|
||||||
|
_attr_state_class = SensorStateClass.MEASUREMENT
|
||||||
|
_attr_should_poll = False
|
||||||
|
|
||||||
|
def __init__(self, serial: str, hub: nobo) -> None:
|
||||||
|
"""Initialize the temperature sensor."""
|
||||||
|
self._temperature: StateType = None
|
||||||
|
self._id = serial
|
||||||
|
self._nobo = hub
|
||||||
|
component = hub.components[self._id]
|
||||||
|
self._attr_unique_id = component[ATTR_SERIAL]
|
||||||
|
self._attr_name = "Temperature"
|
||||||
|
self._attr_has_entity_name = True
|
||||||
|
self._attr_device_info: DeviceInfo = {
|
||||||
|
ATTR_IDENTIFIERS: {(DOMAIN, component[ATTR_SERIAL])},
|
||||||
|
ATTR_NAME: component[ATTR_NAME],
|
||||||
|
ATTR_MANUFACTURER: NOBO_MANUFACTURER,
|
||||||
|
ATTR_MODEL: component[ATTR_MODEL].name,
|
||||||
|
ATTR_VIA_DEVICE: (DOMAIN, hub.hub_info[ATTR_SERIAL]),
|
||||||
|
}
|
||||||
|
zone_id = component[ATTR_ZONE_ID]
|
||||||
|
if zone_id != "-1":
|
||||||
|
self._attr_device_info[ATTR_SUGGESTED_AREA] = hub.zones[zone_id][ATTR_NAME]
|
||||||
|
self._read_state()
|
||||||
|
|
||||||
|
async def async_added_to_hass(self) -> None:
|
||||||
|
"""Register callback from hub."""
|
||||||
|
self._nobo.register_callback(self._after_update)
|
||||||
|
|
||||||
|
async def async_will_remove_from_hass(self) -> None:
|
||||||
|
"""Deregister callback from hub."""
|
||||||
|
self._nobo.deregister_callback(self._after_update)
|
||||||
|
|
||||||
|
@callback
|
||||||
|
def _read_state(self) -> None:
|
||||||
|
"""Read the current state from the hub. This is a local call."""
|
||||||
|
value = self._nobo.get_current_component_temperature(self._id)
|
||||||
|
if value is None:
|
||||||
|
self._attr_native_value = None
|
||||||
|
else:
|
||||||
|
self._attr_native_value = round(float(value), 1)
|
||||||
|
|
||||||
|
@callback
|
||||||
|
def _after_update(self, hub) -> None:
|
||||||
|
self._read_state()
|
||||||
|
self.async_write_ha_state()
|
@ -1742,7 +1742,7 @@ pynetio==0.1.9.1
|
|||||||
pynina==0.1.8
|
pynina==0.1.8
|
||||||
|
|
||||||
# homeassistant.components.nobo_hub
|
# homeassistant.components.nobo_hub
|
||||||
pynobo==1.4.0
|
pynobo==1.6.0
|
||||||
|
|
||||||
# homeassistant.components.nuki
|
# homeassistant.components.nuki
|
||||||
pynuki==1.5.2
|
pynuki==1.5.2
|
||||||
|
@ -1228,7 +1228,7 @@ pynetgear==0.10.8
|
|||||||
pynina==0.1.8
|
pynina==0.1.8
|
||||||
|
|
||||||
# homeassistant.components.nobo_hub
|
# homeassistant.components.nobo_hub
|
||||||
pynobo==1.4.0
|
pynobo==1.6.0
|
||||||
|
|
||||||
# homeassistant.components.nuki
|
# homeassistant.components.nuki
|
||||||
pynuki==1.5.2
|
pynuki==1.5.2
|
||||||
|
Loading…
x
Reference in New Issue
Block a user