Make homekit_controller a local push integration (#32213)
This commit is contained in:
parent
c92aa30663
commit
853d6cda25
5 changed files with 89 additions and 2 deletions
|
@ -46,10 +46,12 @@ class HomeKitEntity(Entity):
|
||||||
)
|
)
|
||||||
|
|
||||||
self._accessory.add_pollable_characteristics(self.pollable_characteristics)
|
self._accessory.add_pollable_characteristics(self.pollable_characteristics)
|
||||||
|
self._accessory.add_watchable_characteristics(self.watchable_characteristics)
|
||||||
|
|
||||||
async def async_will_remove_from_hass(self):
|
async def async_will_remove_from_hass(self):
|
||||||
"""Prepare to be removed from hass."""
|
"""Prepare to be removed from hass."""
|
||||||
self._accessory.remove_pollable_characteristics(self._aid)
|
self._accessory.remove_pollable_characteristics(self._aid)
|
||||||
|
self._accessory.remove_watchable_characteristics(self._aid)
|
||||||
|
|
||||||
for signal_remove in self._signals:
|
for signal_remove in self._signals:
|
||||||
signal_remove()
|
signal_remove()
|
||||||
|
@ -71,6 +73,7 @@ class HomeKitEntity(Entity):
|
||||||
characteristic_types = [get_uuid(c) for c in self.get_characteristic_types()]
|
characteristic_types = [get_uuid(c) for c in self.get_characteristic_types()]
|
||||||
|
|
||||||
self.pollable_characteristics = []
|
self.pollable_characteristics = []
|
||||||
|
self.watchable_characteristics = []
|
||||||
self._chars = {}
|
self._chars = {}
|
||||||
self._char_names = {}
|
self._char_names = {}
|
||||||
|
|
||||||
|
@ -98,6 +101,10 @@ class HomeKitEntity(Entity):
|
||||||
if "pr" in char["perms"]:
|
if "pr" in char["perms"]:
|
||||||
self.pollable_characteristics.append((self._aid, char["iid"]))
|
self.pollable_characteristics.append((self._aid, char["iid"]))
|
||||||
|
|
||||||
|
# Build up a list of (aid, iid) tuples to subscribe to
|
||||||
|
if "ev" in char["perms"]:
|
||||||
|
self.watchable_characteristics.append((self._aid, char["iid"]))
|
||||||
|
|
||||||
# Build a map of ctype -> iid
|
# Build a map of ctype -> iid
|
||||||
short_name = CharacteristicsTypes.get_short(char["type"])
|
short_name = CharacteristicsTypes.get_short(char["type"])
|
||||||
self._chars[short_name] = char["iid"]
|
self._chars[short_name] = char["iid"]
|
||||||
|
|
|
@ -81,7 +81,7 @@ class HomekitControllerFlowHandler(config_entries.ConfigFlow):
|
||||||
"""Handle a HomeKit config flow."""
|
"""Handle a HomeKit config flow."""
|
||||||
|
|
||||||
VERSION = 1
|
VERSION = 1
|
||||||
CONNECTION_CLASS = config_entries.CONN_CLASS_LOCAL_POLL
|
CONNECTION_CLASS = config_entries.CONN_CLASS_LOCAL_PUSH
|
||||||
|
|
||||||
def __init__(self):
|
def __init__(self):
|
||||||
"""Initialize the homekit_controller flow."""
|
"""Initialize the homekit_controller flow."""
|
||||||
|
|
|
@ -108,6 +108,10 @@ class HKDevice:
|
||||||
self._polling_lock = asyncio.Lock()
|
self._polling_lock = asyncio.Lock()
|
||||||
self._polling_lock_warned = False
|
self._polling_lock_warned = False
|
||||||
|
|
||||||
|
self.watchable_characteristics = []
|
||||||
|
|
||||||
|
self.pairing.dispatcher_connect(self.process_new_events)
|
||||||
|
|
||||||
def add_pollable_characteristics(self, characteristics):
|
def add_pollable_characteristics(self, characteristics):
|
||||||
"""Add (aid, iid) pairs that we need to poll."""
|
"""Add (aid, iid) pairs that we need to poll."""
|
||||||
self.pollable_characteristics.extend(characteristics)
|
self.pollable_characteristics.extend(characteristics)
|
||||||
|
@ -118,6 +122,17 @@ class HKDevice:
|
||||||
char for char in self.pollable_characteristics if char[0] != accessory_id
|
char for char in self.pollable_characteristics if char[0] != accessory_id
|
||||||
]
|
]
|
||||||
|
|
||||||
|
def add_watchable_characteristics(self, characteristics):
|
||||||
|
"""Add (aid, iid) pairs that we need to poll."""
|
||||||
|
self.watchable_characteristics.extend(characteristics)
|
||||||
|
self.hass.add_job(self.pairing.subscribe(characteristics))
|
||||||
|
|
||||||
|
def remove_watchable_characteristics(self, accessory_id):
|
||||||
|
"""Remove all pollable characteristics by accessory id."""
|
||||||
|
self.watchable_characteristics = [
|
||||||
|
char for char in self.watchable_characteristics if char[0] != accessory_id
|
||||||
|
]
|
||||||
|
|
||||||
@callback
|
@callback
|
||||||
def async_set_unavailable(self):
|
def async_set_unavailable(self):
|
||||||
"""Mark state of all entities on this connection as unavailable."""
|
"""Mark state of all entities on this connection as unavailable."""
|
||||||
|
@ -163,6 +178,9 @@ class HKDevice:
|
||||||
|
|
||||||
self.add_entities()
|
self.add_entities()
|
||||||
|
|
||||||
|
if self.watchable_characteristics:
|
||||||
|
await self.pairing.subscribe(self.watchable_characteristics)
|
||||||
|
|
||||||
await self.async_update()
|
await self.async_update()
|
||||||
|
|
||||||
return True
|
return True
|
||||||
|
@ -172,6 +190,8 @@ class HKDevice:
|
||||||
if self._polling_interval_remover:
|
if self._polling_interval_remover:
|
||||||
self._polling_interval_remover()
|
self._polling_interval_remover()
|
||||||
|
|
||||||
|
await self.pairing.unsubscribe(self.watchable_characteristics)
|
||||||
|
|
||||||
unloads = []
|
unloads = []
|
||||||
for platform in self.platforms:
|
for platform in self.platforms:
|
||||||
unloads.append(
|
unloads.append(
|
||||||
|
|
|
@ -40,6 +40,11 @@ class Helper:
|
||||||
char_name = CharacteristicsTypes.get_short(char.type)
|
char_name = CharacteristicsTypes.get_short(char.type)
|
||||||
self.characteristics[(service_name, char_name)] = char
|
self.characteristics[(service_name, char_name)] = char
|
||||||
|
|
||||||
|
async def update_named_service(self, service, characteristics):
|
||||||
|
"""Update a service."""
|
||||||
|
self.pairing.testing.update_named_service(service, characteristics)
|
||||||
|
await self.hass.async_block_till_done()
|
||||||
|
|
||||||
async def poll_and_get_state(self):
|
async def poll_and_get_state(self):
|
||||||
"""Trigger a time based poll and return the current entity state."""
|
"""Trigger a time based poll and return the current entity state."""
|
||||||
await time_changed(self.hass, 60)
|
await time_changed(self.hass, 60)
|
||||||
|
|
|
@ -6,6 +6,9 @@ from homeassistant.components.homekit_controller.const import KNOWN_DEVICES
|
||||||
|
|
||||||
from tests.components.homekit_controller.common import setup_test_component
|
from tests.components.homekit_controller.common import setup_test_component
|
||||||
|
|
||||||
|
LIGHT_BULB_NAME = "Light Bulb"
|
||||||
|
LIGHT_BULB_ENTITY_ID = "light.testdevice"
|
||||||
|
|
||||||
LIGHT_ON = ("lightbulb", "on")
|
LIGHT_ON = ("lightbulb", "on")
|
||||||
LIGHT_BRIGHTNESS = ("lightbulb", "brightness")
|
LIGHT_BRIGHTNESS = ("lightbulb", "brightness")
|
||||||
LIGHT_HUE = ("lightbulb", "hue")
|
LIGHT_HUE = ("lightbulb", "hue")
|
||||||
|
@ -15,7 +18,7 @@ LIGHT_COLOR_TEMP = ("lightbulb", "color-temperature")
|
||||||
|
|
||||||
def create_lightbulb_service(accessory):
|
def create_lightbulb_service(accessory):
|
||||||
"""Define lightbulb characteristics."""
|
"""Define lightbulb characteristics."""
|
||||||
service = accessory.add_service(ServicesTypes.LIGHTBULB)
|
service = accessory.add_service(ServicesTypes.LIGHTBULB, name=LIGHT_BULB_NAME)
|
||||||
|
|
||||||
on_char = service.add_char(CharacteristicsTypes.ON)
|
on_char = service.add_char(CharacteristicsTypes.ON)
|
||||||
on_char.value = 0
|
on_char.value = 0
|
||||||
|
@ -110,6 +113,35 @@ async def test_switch_read_light_state(hass, utcnow):
|
||||||
assert state.state == "off"
|
assert state.state == "off"
|
||||||
|
|
||||||
|
|
||||||
|
async def test_switch_push_light_state(hass, utcnow):
|
||||||
|
"""Test that we can read the state of a HomeKit light accessory."""
|
||||||
|
helper = await setup_test_component(hass, create_lightbulb_service_with_hs)
|
||||||
|
|
||||||
|
# Initial state is that the light is off
|
||||||
|
state = hass.states.get(LIGHT_BULB_ENTITY_ID)
|
||||||
|
assert state.state == "off"
|
||||||
|
|
||||||
|
await helper.update_named_service(
|
||||||
|
LIGHT_BULB_NAME,
|
||||||
|
{
|
||||||
|
CharacteristicsTypes.ON: True,
|
||||||
|
CharacteristicsTypes.BRIGHTNESS: 100,
|
||||||
|
CharacteristicsTypes.HUE: 4,
|
||||||
|
CharacteristicsTypes.SATURATION: 5,
|
||||||
|
},
|
||||||
|
)
|
||||||
|
|
||||||
|
state = hass.states.get(LIGHT_BULB_ENTITY_ID)
|
||||||
|
assert state.state == "on"
|
||||||
|
assert state.attributes["brightness"] == 255
|
||||||
|
assert state.attributes["hs_color"] == (4, 5)
|
||||||
|
|
||||||
|
# Simulate that device switched off in the real world not via HA
|
||||||
|
await helper.update_named_service(LIGHT_BULB_NAME, {CharacteristicsTypes.ON: False})
|
||||||
|
state = hass.states.get(LIGHT_BULB_ENTITY_ID)
|
||||||
|
assert state.state == "off"
|
||||||
|
|
||||||
|
|
||||||
async def test_switch_read_light_state_color_temp(hass, utcnow):
|
async def test_switch_read_light_state_color_temp(hass, utcnow):
|
||||||
"""Test that we can read the color_temp of a light accessory."""
|
"""Test that we can read the color_temp of a light accessory."""
|
||||||
helper = await setup_test_component(hass, create_lightbulb_service_with_color_temp)
|
helper = await setup_test_component(hass, create_lightbulb_service_with_color_temp)
|
||||||
|
@ -129,6 +161,29 @@ async def test_switch_read_light_state_color_temp(hass, utcnow):
|
||||||
assert state.attributes["color_temp"] == 400
|
assert state.attributes["color_temp"] == 400
|
||||||
|
|
||||||
|
|
||||||
|
async def test_switch_push_light_state_color_temp(hass, utcnow):
|
||||||
|
"""Test that we can read the state of a HomeKit light accessory."""
|
||||||
|
helper = await setup_test_component(hass, create_lightbulb_service_with_color_temp)
|
||||||
|
|
||||||
|
# Initial state is that the light is off
|
||||||
|
state = hass.states.get(LIGHT_BULB_ENTITY_ID)
|
||||||
|
assert state.state == "off"
|
||||||
|
|
||||||
|
await helper.update_named_service(
|
||||||
|
LIGHT_BULB_NAME,
|
||||||
|
{
|
||||||
|
CharacteristicsTypes.ON: True,
|
||||||
|
CharacteristicsTypes.BRIGHTNESS: 100,
|
||||||
|
CharacteristicsTypes.COLOR_TEMPERATURE: 400,
|
||||||
|
},
|
||||||
|
)
|
||||||
|
|
||||||
|
state = hass.states.get(LIGHT_BULB_ENTITY_ID)
|
||||||
|
assert state.state == "on"
|
||||||
|
assert state.attributes["brightness"] == 255
|
||||||
|
assert state.attributes["color_temp"] == 400
|
||||||
|
|
||||||
|
|
||||||
async def test_light_becomes_unavailable_but_recovers(hass, utcnow):
|
async def test_light_becomes_unavailable_but_recovers(hass, utcnow):
|
||||||
"""Test transition to and from unavailable state."""
|
"""Test transition to and from unavailable state."""
|
||||||
helper = await setup_test_component(hass, create_lightbulb_service_with_color_temp)
|
helper = await setup_test_component(hass, create_lightbulb_service_with_color_temp)
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue