Skip to content

Commit ee8e7b9

Browse files
committed
Get status once per device instead of per platform
1 parent 208b0ad commit ee8e7b9

File tree

8 files changed

+76
-29
lines changed

8 files changed

+76
-29
lines changed

custom_components/localtuya/__init__.py

+39-5
Original file line numberDiff line numberDiff line change
@@ -48,13 +48,17 @@
4848
"""
4949
import asyncio
5050
import logging
51+
from datetime import timedelta, datetime
5152

5253
from homeassistant.config_entries import SOURCE_IMPORT, ConfigEntry
5354
from homeassistant.core import HomeAssistant
5455
from homeassistant.const import (
56+
CONF_DEVICE_ID,
5557
CONF_PLATFORM,
5658
CONF_ENTITIES,
5759
)
60+
from homeassistant.helpers.event import async_track_time_interval
61+
from homeassistant.helpers.dispatcher import async_dispatcher_send
5862

5963
from .const import DOMAIN, TUYA_DEVICE
6064
from .config_flow import config_schema
@@ -63,6 +67,9 @@
6367
_LOGGER = logging.getLogger(__name__)
6468

6569
UNSUB_LISTENER = "unsub_listener"
70+
UNSUB_TRACK = "unsub_track"
71+
72+
POLL_INTERVAL = 30
6673

6774
CONFIG_SCHEMA = config_schema()
6875

@@ -85,17 +92,43 @@ async def async_setup_entry(hass: HomeAssistant, entry: ConfigEntry):
8592
"""Set up LocalTuya integration from a config entry."""
8693
unsub_listener = entry.add_update_listener(update_listener)
8794

95+
device = TuyaDevice(entry.data)
96+
97+
def update_state(now):
98+
"""Read device status and update platforms."""
99+
status = None
100+
try:
101+
status = device.status()
102+
except Exception:
103+
_LOGGER.exception("update failed")
104+
105+
signal = f"localtuya_{entry.data[CONF_DEVICE_ID]}"
106+
async_dispatcher_send(hass, signal, status)
107+
108+
unsub_track = async_track_time_interval(
109+
hass, update_state, timedelta(seconds=POLL_INTERVAL)
110+
)
111+
88112
hass.data[DOMAIN][entry.entry_id] = {
89113
UNSUB_LISTENER: unsub_listener,
90-
TUYA_DEVICE: TuyaDevice(entry.data),
114+
UNSUB_TRACK: unsub_track,
115+
TUYA_DEVICE: device,
91116
}
92117

93-
for entity in entry.data[CONF_ENTITIES]:
94-
platform = entity[CONF_PLATFORM]
95-
hass.async_create_task(
96-
hass.config_entries.async_forward_entry_setup(entry, platform)
118+
async def setup_entities():
119+
await asyncio.gather(
120+
*[
121+
hass.config_entries.async_forward_entry_setup(
122+
entry, entity[CONF_PLATFORM]
123+
)
124+
for entity in entry.data[CONF_ENTITIES]
125+
]
97126
)
98127

128+
update_state(datetime.now())
129+
130+
hass.async_create_task(setup_entities())
131+
99132
return True
100133

101134

@@ -113,6 +146,7 @@ async def async_unload_entry(hass: HomeAssistant, entry: ConfigEntry):
113146
)
114147

115148
hass.data[DOMAIN][entry.entry_id][UNSUB_LISTENER]()
149+
hass.data[DOMAIN][entry.entry_id][UNSUB_TRACK]()
116150
if unload_ok:
117151
hass.data[DOMAIN].pop(entry.entry_id)
118152

custom_components/localtuya/binary_sensor.py

+1-1
Original file line numberDiff line numberDiff line change
@@ -45,7 +45,7 @@ async def async_setup_entry(hass, config_entry, async_add_entities):
4545
)
4646
)
4747

48-
async_add_entities(sensors, True)
48+
async_add_entities(sensors)
4949

5050

5151
class LocaltuyaBinarySensor(LocalTuyaEntity, BinarySensorEntity):

custom_components/localtuya/common.py

+31-13
Original file line numberDiff line numberDiff line change
@@ -4,6 +4,7 @@
44
from threading import Lock
55

66
from homeassistant.helpers.entity import Entity
7+
from homeassistant.helpers.dispatcher import async_dispatcher_connect
78

89
from homeassistant.const import (
910
CONF_DEVICE_ID,
@@ -132,9 +133,27 @@ def __init__(self, device, config_entry, dps_id, **kwargs):
132133
self._device = device
133134
self._config_entry = config_entry
134135
self._config = get_entity_config(config_entry, dps_id)
135-
self._available = False
136136
self._dps_id = dps_id
137-
self._status = None
137+
self._status = {}
138+
139+
async def async_added_to_hass(self):
140+
"""Subscribe localtuya events."""
141+
await super().async_added_to_hass()
142+
143+
def _update_handler(status):
144+
"""Update entity state when status was updated."""
145+
if status is not None:
146+
self._status = status
147+
self.status_updated()
148+
else:
149+
self._status = {}
150+
151+
self.schedule_update_ha_state()
152+
153+
signal = f"localtuya_{self._config_entry.data[CONF_DEVICE_ID]}"
154+
self.async_on_remove(
155+
async_dispatcher_connect(self.hass, signal, _update_handler)
156+
)
138157

139158
@property
140159
def device_info(self):
@@ -155,6 +174,11 @@ def name(self):
155174
"""Get name of Tuya entity."""
156175
return self._config[CONF_FRIENDLY_NAME]
157176

177+
@property
178+
def should_poll(self):
179+
"""Return if platform should poll for updates."""
180+
return False
181+
158182
@property
159183
def unique_id(self):
160184
"""Return unique device identifier."""
@@ -163,28 +187,22 @@ def unique_id(self):
163187
@property
164188
def available(self):
165189
"""Return if device is available or not."""
166-
return self._available
190+
return bool(self._status)
167191

168192
def dps(self, dps_index):
169193
"""Return cached value for DPS index."""
194+
if "dps" not in self._status:
195+
return None
196+
170197
value = self._status["dps"].get(str(dps_index))
171198
if value is None:
172199
_LOGGER.warning(
173200
"Entity %s is requesting unknown DPS index %s",
174201
self.entity_id,
175202
dps_index,
176203
)
177-
return value
178204

179-
def update(self):
180-
"""Update state of Tuya entity."""
181-
try:
182-
self._status = self._device.status()
183-
self.status_updated()
184-
except Exception:
185-
self._available = False
186-
else:
187-
self._available = True
205+
return value
188206

189207
def status_updated(self):
190208
"""Device status was updated.

custom_components/localtuya/cover.py

+1-6
Original file line numberDiff line numberDiff line change
@@ -55,7 +55,7 @@ async def async_setup_entry(hass, config_entry, async_add_entities):
5555
)
5656
)
5757

58-
async_add_entities(covers, True)
58+
async_add_entities(covers)
5959

6060

6161
class LocaltuyaCover(LocalTuyaEntity, CoverEntity):
@@ -78,11 +78,6 @@ def __init__(
7878
)
7979
)
8080

81-
@property
82-
def available(self):
83-
"""Return if device is available or not."""
84-
return self._available
85-
8681
@property
8782
def supported_features(self):
8883
"""Flag supported features."""

custom_components/localtuya/fan.py

+1-1
Original file line numberDiff line numberDiff line change
@@ -42,7 +42,7 @@ async def async_setup_entry(hass, config_entry, async_add_entities):
4242
)
4343
)
4444

45-
async_add_entities(fans, True)
45+
async_add_entities(fans)
4646

4747

4848
class LocaltuyaFan(LocalTuyaEntity, FanEntity):

custom_components/localtuya/light.py

+1-1
Original file line numberDiff line numberDiff line change
@@ -50,7 +50,7 @@ async def async_setup_entry(hass, config_entry, async_add_entities):
5050
)
5151
)
5252

53-
async_add_entities(lights, True)
53+
async_add_entities(lights)
5454

5555

5656
class LocaltuyaLight(LocalTuyaEntity, LightEntity):

custom_components/localtuya/sensor.py

+1-1
Original file line numberDiff line numberDiff line change
@@ -49,7 +49,7 @@ async def async_setup_entry(hass, config_entry, async_add_entities):
4949
)
5050
)
5151

52-
async_add_entities(sensors, True)
52+
async_add_entities(sensors)
5353

5454

5555
class LocaltuyaSensor(LocalTuyaEntity):

custom_components/localtuya/switch.py

+1-1
Original file line numberDiff line numberDiff line change
@@ -49,7 +49,7 @@ async def async_setup_entry(hass, config_entry, async_add_entities):
4949
)
5050
)
5151

52-
async_add_entities(switches, True)
52+
async_add_entities(switches)
5353

5454

5555
class LocaltuyaSwitch(LocalTuyaEntity, SwitchEntity):

0 commit comments

Comments
 (0)