Improve NetAtmo sensors update logic (#14866)
* Added a "last update" sensor that could be used by automations + cosmetic changes * Improved the update logic of sensor data The platform is now continuously adjusting the refresh interval in order to synchronize with the expected next update from the NetAtmo cloud. This significantly improves reaction time of automations while keeping the refresh time to the recommended value (10 minutes). * Linting * Incorporated the advanced Throttle class to support adaptive throttling, as opposed to integrating it in the core framework. Following code review, it was suggested to implement the specialised Throttle class in this platform instead of making a change in the general util package. Except that the required change (about 4 LoC) is part of the only relevant piece of code of that class, therefore this commit includes a full copy of the Throttle class from homeassistant.util, plus the extra feature to support adaptive throttling. * Cosmetic changes on the introduced "last updated" sensor * Alternate implementation for the adaptive throttling Ensure the updates from the cloud are throttled and adapted to the last update time provided by NetAtmo, without using the Throttle decorator. Similar logic and similar usage of a lock to protect the execution of the remote update. * Lintingpull/15404/head
parent
2ee62b10bc
commit
9ea0c409e6
|
@ -5,7 +5,8 @@ For more details about this platform, please refer to the documentation at
|
||||||
https://home-assistant.io/components/sensor.netatmo/
|
https://home-assistant.io/components/sensor.netatmo/
|
||||||
"""
|
"""
|
||||||
import logging
|
import logging
|
||||||
from datetime import timedelta
|
from time import time
|
||||||
|
import threading
|
||||||
|
|
||||||
import voluptuous as vol
|
import voluptuous as vol
|
||||||
|
|
||||||
|
@ -14,7 +15,6 @@ from homeassistant.const import (
|
||||||
TEMP_CELSIUS, DEVICE_CLASS_HUMIDITY, DEVICE_CLASS_TEMPERATURE,
|
TEMP_CELSIUS, DEVICE_CLASS_HUMIDITY, DEVICE_CLASS_TEMPERATURE,
|
||||||
STATE_UNKNOWN)
|
STATE_UNKNOWN)
|
||||||
from homeassistant.helpers.entity import Entity
|
from homeassistant.helpers.entity import Entity
|
||||||
from homeassistant.util import Throttle
|
|
||||||
import homeassistant.helpers.config_validation as cv
|
import homeassistant.helpers.config_validation as cv
|
||||||
|
|
||||||
_LOGGER = logging.getLogger(__name__)
|
_LOGGER = logging.getLogger(__name__)
|
||||||
|
@ -24,8 +24,8 @@ CONF_STATION = 'station'
|
||||||
|
|
||||||
DEPENDENCIES = ['netatmo']
|
DEPENDENCIES = ['netatmo']
|
||||||
|
|
||||||
# NetAtmo Data is uploaded to server every 10 minutes
|
# This is the NetAtmo data upload interval in seconds
|
||||||
MIN_TIME_BETWEEN_UPDATES = timedelta(seconds=600)
|
NETATMO_UPDATE_INTERVAL = 600
|
||||||
|
|
||||||
SENSOR_TYPES = {
|
SENSOR_TYPES = {
|
||||||
'temperature': ['Temperature', TEMP_CELSIUS, None,
|
'temperature': ['Temperature', TEMP_CELSIUS, None,
|
||||||
|
@ -50,7 +50,8 @@ SENSOR_TYPES = {
|
||||||
'rf_status': ['Radio', '', 'mdi:signal', None],
|
'rf_status': ['Radio', '', 'mdi:signal', None],
|
||||||
'rf_status_lvl': ['Radio_lvl', '', 'mdi:signal', None],
|
'rf_status_lvl': ['Radio_lvl', '', 'mdi:signal', None],
|
||||||
'wifi_status': ['Wifi', '', 'mdi:wifi', None],
|
'wifi_status': ['Wifi', '', 'mdi:wifi', None],
|
||||||
'wifi_status_lvl': ['Wifi_lvl', 'dBm', 'mdi:wifi', None]
|
'wifi_status_lvl': ['Wifi_lvl', 'dBm', 'mdi:wifi', None],
|
||||||
|
'lastupdated': ['Last Updated', 's', 'mdi:timer', None],
|
||||||
}
|
}
|
||||||
|
|
||||||
MODULE_SCHEMA = vol.Schema({
|
MODULE_SCHEMA = vol.Schema({
|
||||||
|
@ -76,11 +77,11 @@ def setup_platform(hass, config, add_devices, discovery_info=None):
|
||||||
# Iterate each module
|
# Iterate each module
|
||||||
for module_name, monitored_conditions in\
|
for module_name, monitored_conditions in\
|
||||||
config[CONF_MODULES].items():
|
config[CONF_MODULES].items():
|
||||||
# Test if module exist """
|
# Test if module exists
|
||||||
if module_name not in data.get_module_names():
|
if module_name not in data.get_module_names():
|
||||||
_LOGGER.error('Module name: "%s" not found', module_name)
|
_LOGGER.error('Module name: "%s" not found', module_name)
|
||||||
continue
|
continue
|
||||||
# Only create sensor for monitored """
|
# Only create sensors for monitored properties
|
||||||
for variable in monitored_conditions:
|
for variable in monitored_conditions:
|
||||||
dev.append(NetAtmoSensor(data, module_name, variable))
|
dev.append(NetAtmoSensor(data, module_name, variable))
|
||||||
else:
|
else:
|
||||||
|
@ -285,6 +286,8 @@ class NetAtmoSensor(Entity):
|
||||||
self._state = "High"
|
self._state = "High"
|
||||||
elif data['wifi_status'] <= 55:
|
elif data['wifi_status'] <= 55:
|
||||||
self._state = "Full"
|
self._state = "Full"
|
||||||
|
elif self.type == 'lastupdated':
|
||||||
|
self._state = int(time() - data['When'])
|
||||||
|
|
||||||
|
|
||||||
class NetAtmoData(object):
|
class NetAtmoData(object):
|
||||||
|
@ -296,20 +299,57 @@ class NetAtmoData(object):
|
||||||
self.data = None
|
self.data = None
|
||||||
self.station_data = None
|
self.station_data = None
|
||||||
self.station = station
|
self.station = station
|
||||||
|
self._next_update = time()
|
||||||
|
self._update_in_progress = threading.Lock()
|
||||||
|
|
||||||
def get_module_names(self):
|
def get_module_names(self):
|
||||||
"""Return all module available on the API as a list."""
|
"""Return all module available on the API as a list."""
|
||||||
self.update()
|
self.update()
|
||||||
return self.data.keys()
|
return self.data.keys()
|
||||||
|
|
||||||
@Throttle(MIN_TIME_BETWEEN_UPDATES)
|
|
||||||
def update(self):
|
def update(self):
|
||||||
"""Call the Netatmo API to update the data."""
|
"""Call the Netatmo API to update the data.
|
||||||
import pyatmo
|
|
||||||
self.station_data = pyatmo.WeatherStationData(self.auth)
|
|
||||||
|
|
||||||
if self.station is not None:
|
This method is not throttled by the builtin Throttle decorator
|
||||||
self.data = self.station_data.lastData(
|
but with a custom logic, which takes into account the time
|
||||||
station=self.station, exclude=3600)
|
of the last update from the cloud.
|
||||||
else:
|
"""
|
||||||
self.data = self.station_data.lastData(exclude=3600)
|
if time() < self._next_update or \
|
||||||
|
not self._update_in_progress.acquire(False):
|
||||||
|
return
|
||||||
|
|
||||||
|
try:
|
||||||
|
import pyatmo
|
||||||
|
self.station_data = pyatmo.WeatherStationData(self.auth)
|
||||||
|
|
||||||
|
if self.station is not None:
|
||||||
|
self.data = self.station_data.lastData(
|
||||||
|
station=self.station, exclude=3600)
|
||||||
|
else:
|
||||||
|
self.data = self.station_data.lastData(exclude=3600)
|
||||||
|
|
||||||
|
newinterval = 0
|
||||||
|
for module in self.data:
|
||||||
|
if 'When' in self.data[module]:
|
||||||
|
newinterval = self.data[module]['When']
|
||||||
|
break
|
||||||
|
if newinterval:
|
||||||
|
# Try and estimate when fresh data will be available
|
||||||
|
newinterval += NETATMO_UPDATE_INTERVAL - time()
|
||||||
|
if newinterval > NETATMO_UPDATE_INTERVAL - 30:
|
||||||
|
newinterval = NETATMO_UPDATE_INTERVAL
|
||||||
|
else:
|
||||||
|
if newinterval < NETATMO_UPDATE_INTERVAL / 2:
|
||||||
|
# Never hammer the NetAtmo API more than
|
||||||
|
# twice per update interval
|
||||||
|
newinterval = NETATMO_UPDATE_INTERVAL / 2
|
||||||
|
_LOGGER.warning(
|
||||||
|
"NetAtmo refresh interval reset to %d seconds",
|
||||||
|
newinterval)
|
||||||
|
else:
|
||||||
|
# Last update time not found, fall back to default value
|
||||||
|
newinterval = NETATMO_UPDATE_INTERVAL
|
||||||
|
|
||||||
|
self._next_update = time() + newinterval
|
||||||
|
finally:
|
||||||
|
self._update_in_progress.release()
|
||||||
|
|
Loading…
Reference in New Issue