mirror of
https://github.com/home-assistant/core.git
synced 2025-07-22 20:57:21 +00:00
Add fingerprint and nfc event support to unifiprotect (#130840)
Co-authored-by: J. Nick Koston <nick@koston.org>
This commit is contained in:
parent
521cc67d45
commit
91e4939bf0
@ -1,5 +1,7 @@
|
|||||||
"""Constant definitions for UniFi Protect Integration."""
|
"""Constant definitions for UniFi Protect Integration."""
|
||||||
|
|
||||||
|
from typing import Final
|
||||||
|
|
||||||
from uiprotect.data import ModelType, Version
|
from uiprotect.data import ModelType, Version
|
||||||
|
|
||||||
from homeassistant.const import Platform
|
from homeassistant.const import Platform
|
||||||
@ -75,3 +77,8 @@ PLATFORMS = [
|
|||||||
DISPATCH_ADD = "add_device"
|
DISPATCH_ADD = "add_device"
|
||||||
DISPATCH_ADOPT = "adopt_device"
|
DISPATCH_ADOPT = "adopt_device"
|
||||||
DISPATCH_CHANNELS = "new_camera_channels"
|
DISPATCH_CHANNELS = "new_camera_channels"
|
||||||
|
|
||||||
|
EVENT_TYPE_FINGERPRINT_IDENTIFIED: Final = "identified"
|
||||||
|
EVENT_TYPE_FINGERPRINT_NOT_IDENTIFIED: Final = "not_identified"
|
||||||
|
EVENT_TYPE_NFC_SCANNED: Final = "scanned"
|
||||||
|
EVENT_TYPE_DOORBELL_RING: Final = "ring"
|
||||||
|
@ -14,7 +14,13 @@ from homeassistant.components.event import (
|
|||||||
from homeassistant.core import HomeAssistant, callback
|
from homeassistant.core import HomeAssistant, callback
|
||||||
from homeassistant.helpers.entity_platform import AddEntitiesCallback
|
from homeassistant.helpers.entity_platform import AddEntitiesCallback
|
||||||
|
|
||||||
from .const import ATTR_EVENT_ID
|
from .const import (
|
||||||
|
ATTR_EVENT_ID,
|
||||||
|
EVENT_TYPE_DOORBELL_RING,
|
||||||
|
EVENT_TYPE_FINGERPRINT_IDENTIFIED,
|
||||||
|
EVENT_TYPE_FINGERPRINT_NOT_IDENTIFIED,
|
||||||
|
EVENT_TYPE_NFC_SCANNED,
|
||||||
|
)
|
||||||
from .data import ProtectData, ProtectDeviceType, UFPConfigEntry
|
from .data import ProtectData, ProtectDeviceType, UFPConfigEntry
|
||||||
from .entity import EventEntityMixin, ProtectDeviceEntity, ProtectEventMixin
|
from .entity import EventEntityMixin, ProtectDeviceEntity, ProtectEventMixin
|
||||||
|
|
||||||
@ -23,22 +29,10 @@ from .entity import EventEntityMixin, ProtectDeviceEntity, ProtectEventMixin
|
|||||||
class ProtectEventEntityDescription(ProtectEventMixin, EventEntityDescription):
|
class ProtectEventEntityDescription(ProtectEventMixin, EventEntityDescription):
|
||||||
"""Describes UniFi Protect event entity."""
|
"""Describes UniFi Protect event entity."""
|
||||||
|
|
||||||
|
entity_class: type[ProtectDeviceEntity]
|
||||||
EVENT_DESCRIPTIONS: tuple[ProtectEventEntityDescription, ...] = (
|
|
||||||
ProtectEventEntityDescription(
|
|
||||||
key="doorbell",
|
|
||||||
translation_key="doorbell",
|
|
||||||
name="Doorbell",
|
|
||||||
device_class=EventDeviceClass.DOORBELL,
|
|
||||||
icon="mdi:doorbell-video",
|
|
||||||
ufp_required_field="feature_flags.is_doorbell",
|
|
||||||
ufp_event_obj="last_ring_event",
|
|
||||||
event_types=[EventType.RING],
|
|
||||||
),
|
|
||||||
)
|
|
||||||
|
|
||||||
|
|
||||||
class ProtectDeviceEventEntity(EventEntityMixin, ProtectDeviceEntity, EventEntity):
|
class ProtectDeviceRingEventEntity(EventEntityMixin, ProtectDeviceEntity, EventEntity):
|
||||||
"""A UniFi Protect event entity."""
|
"""A UniFi Protect event entity."""
|
||||||
|
|
||||||
entity_description: ProtectEventEntityDescription
|
entity_description: ProtectEventEntityDescription
|
||||||
@ -57,26 +51,128 @@ class ProtectDeviceEventEntity(EventEntityMixin, ProtectDeviceEntity, EventEntit
|
|||||||
if (
|
if (
|
||||||
event
|
event
|
||||||
and not self._event_already_ended(prev_event, prev_event_end)
|
and not self._event_already_ended(prev_event, prev_event_end)
|
||||||
and (event_types := description.event_types)
|
and event.type is EventType.RING
|
||||||
and (event_type := event.type) in event_types
|
|
||||||
):
|
):
|
||||||
self._trigger_event(event_type, {ATTR_EVENT_ID: event.id})
|
self._trigger_event(EVENT_TYPE_DOORBELL_RING, {ATTR_EVENT_ID: event.id})
|
||||||
self.async_write_ha_state()
|
self.async_write_ha_state()
|
||||||
|
|
||||||
|
|
||||||
|
class ProtectDeviceNFCEventEntity(EventEntityMixin, ProtectDeviceEntity, EventEntity):
|
||||||
|
"""A UniFi Protect NFC event entity."""
|
||||||
|
|
||||||
|
entity_description: ProtectEventEntityDescription
|
||||||
|
|
||||||
|
@callback
|
||||||
|
def _async_update_device_from_protect(self, device: ProtectDeviceType) -> None:
|
||||||
|
description = self.entity_description
|
||||||
|
|
||||||
|
prev_event = self._event
|
||||||
|
prev_event_end = self._event_end
|
||||||
|
super()._async_update_device_from_protect(device)
|
||||||
|
if event := description.get_event_obj(device):
|
||||||
|
self._event = event
|
||||||
|
self._event_end = event.end if event else None
|
||||||
|
|
||||||
|
if (
|
||||||
|
event
|
||||||
|
and not self._event_already_ended(prev_event, prev_event_end)
|
||||||
|
and event.type is EventType.NFC_CARD_SCANNED
|
||||||
|
):
|
||||||
|
event_data = {ATTR_EVENT_ID: event.id}
|
||||||
|
if event.metadata and event.metadata.nfc and event.metadata.nfc.nfc_id:
|
||||||
|
event_data["nfc_id"] = event.metadata.nfc.nfc_id
|
||||||
|
|
||||||
|
self._trigger_event(EVENT_TYPE_NFC_SCANNED, event_data)
|
||||||
|
self.async_write_ha_state()
|
||||||
|
|
||||||
|
|
||||||
|
class ProtectDeviceFingerprintEventEntity(
|
||||||
|
EventEntityMixin, ProtectDeviceEntity, EventEntity
|
||||||
|
):
|
||||||
|
"""A UniFi Protect fingerprint event entity."""
|
||||||
|
|
||||||
|
entity_description: ProtectEventEntityDescription
|
||||||
|
|
||||||
|
@callback
|
||||||
|
def _async_update_device_from_protect(self, device: ProtectDeviceType) -> None:
|
||||||
|
description = self.entity_description
|
||||||
|
|
||||||
|
prev_event = self._event
|
||||||
|
prev_event_end = self._event_end
|
||||||
|
super()._async_update_device_from_protect(device)
|
||||||
|
if event := description.get_event_obj(device):
|
||||||
|
self._event = event
|
||||||
|
self._event_end = event.end if event else None
|
||||||
|
|
||||||
|
if (
|
||||||
|
event
|
||||||
|
and not self._event_already_ended(prev_event, prev_event_end)
|
||||||
|
and event.type is EventType.FINGERPRINT_IDENTIFIED
|
||||||
|
):
|
||||||
|
event_data = {ATTR_EVENT_ID: event.id}
|
||||||
|
if (
|
||||||
|
event.metadata
|
||||||
|
and event.metadata.fingerprint
|
||||||
|
and event.metadata.fingerprint.ulp_id
|
||||||
|
):
|
||||||
|
event_data["ulp_id"] = event.metadata.fingerprint.ulp_id
|
||||||
|
event_identified = EVENT_TYPE_FINGERPRINT_IDENTIFIED
|
||||||
|
else:
|
||||||
|
event_data["ulp_id"] = ""
|
||||||
|
event_identified = EVENT_TYPE_FINGERPRINT_NOT_IDENTIFIED
|
||||||
|
|
||||||
|
self._trigger_event(event_identified, event_data)
|
||||||
|
self.async_write_ha_state()
|
||||||
|
|
||||||
|
|
||||||
|
EVENT_DESCRIPTIONS: tuple[ProtectEventEntityDescription, ...] = (
|
||||||
|
ProtectEventEntityDescription(
|
||||||
|
key="doorbell",
|
||||||
|
translation_key="doorbell",
|
||||||
|
device_class=EventDeviceClass.DOORBELL,
|
||||||
|
icon="mdi:doorbell-video",
|
||||||
|
ufp_required_field="feature_flags.is_doorbell",
|
||||||
|
ufp_event_obj="last_ring_event",
|
||||||
|
event_types=[EVENT_TYPE_DOORBELL_RING],
|
||||||
|
entity_class=ProtectDeviceRingEventEntity,
|
||||||
|
),
|
||||||
|
ProtectEventEntityDescription(
|
||||||
|
key="nfc",
|
||||||
|
translation_key="nfc",
|
||||||
|
device_class=EventDeviceClass.DOORBELL,
|
||||||
|
icon="mdi:nfc",
|
||||||
|
ufp_required_field="feature_flags.support_nfc",
|
||||||
|
ufp_event_obj="last_nfc_card_scanned_event",
|
||||||
|
event_types=[EVENT_TYPE_NFC_SCANNED],
|
||||||
|
entity_class=ProtectDeviceNFCEventEntity,
|
||||||
|
),
|
||||||
|
ProtectEventEntityDescription(
|
||||||
|
key="fingerprint",
|
||||||
|
translation_key="fingerprint",
|
||||||
|
device_class=EventDeviceClass.DOORBELL,
|
||||||
|
icon="mdi:fingerprint",
|
||||||
|
ufp_required_field="feature_flags.has_fingerprint_sensor",
|
||||||
|
ufp_event_obj="last_fingerprint_identified_event",
|
||||||
|
event_types=[
|
||||||
|
EVENT_TYPE_FINGERPRINT_IDENTIFIED,
|
||||||
|
EVENT_TYPE_FINGERPRINT_NOT_IDENTIFIED,
|
||||||
|
],
|
||||||
|
entity_class=ProtectDeviceFingerprintEventEntity,
|
||||||
|
),
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
@callback
|
@callback
|
||||||
def _async_event_entities(
|
def _async_event_entities(
|
||||||
data: ProtectData,
|
data: ProtectData,
|
||||||
ufp_device: ProtectAdoptableDeviceModel | None = None,
|
ufp_device: ProtectAdoptableDeviceModel | None = None,
|
||||||
) -> list[ProtectDeviceEntity]:
|
) -> list[ProtectDeviceEntity]:
|
||||||
entities: list[ProtectDeviceEntity] = []
|
return [
|
||||||
for device in data.get_cameras() if ufp_device is None else [ufp_device]:
|
description.entity_class(data, device, description)
|
||||||
entities.extend(
|
for device in (data.get_cameras() if ufp_device is None else [ufp_device])
|
||||||
ProtectDeviceEventEntity(data, device, description)
|
for description in EVENT_DESCRIPTIONS
|
||||||
for description in EVENT_DESCRIPTIONS
|
if description.has_required(device)
|
||||||
if description.has_required(device)
|
]
|
||||||
)
|
|
||||||
return entities
|
|
||||||
|
|
||||||
|
|
||||||
async def async_setup_entry(
|
async def async_setup_entry(
|
||||||
|
@ -137,6 +137,7 @@
|
|||||||
},
|
},
|
||||||
"event": {
|
"event": {
|
||||||
"doorbell": {
|
"doorbell": {
|
||||||
|
"name": "Doorbell",
|
||||||
"state_attributes": {
|
"state_attributes": {
|
||||||
"event_type": {
|
"event_type": {
|
||||||
"state": {
|
"state": {
|
||||||
@ -144,6 +145,27 @@
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
},
|
||||||
|
"nfc": {
|
||||||
|
"name": "NFC",
|
||||||
|
"state_attributes": {
|
||||||
|
"event_type": {
|
||||||
|
"state": {
|
||||||
|
"scanned": "Scanned"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"fingerprint": {
|
||||||
|
"name": "Fingerprint",
|
||||||
|
"state_attributes": {
|
||||||
|
"event_type": {
|
||||||
|
"state": {
|
||||||
|
"identified": "Identified",
|
||||||
|
"not_identified": "Not identified"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
@ -233,6 +233,8 @@ def doorbell_fixture(camera: Camera, fixed_now: datetime):
|
|||||||
doorbell.feature_flags.has_speaker = True
|
doorbell.feature_flags.has_speaker = True
|
||||||
doorbell.feature_flags.has_privacy_mask = True
|
doorbell.feature_flags.has_privacy_mask = True
|
||||||
doorbell.feature_flags.is_doorbell = True
|
doorbell.feature_flags.is_doorbell = True
|
||||||
|
doorbell.feature_flags.has_fingerprint_sensor = True
|
||||||
|
doorbell.feature_flags.support_nfc = True
|
||||||
doorbell.feature_flags.has_chime = True
|
doorbell.feature_flags.has_chime = True
|
||||||
doorbell.feature_flags.has_smart_detect = True
|
doorbell.feature_flags.has_smart_detect = True
|
||||||
doorbell.feature_flags.has_package_camera = True
|
doorbell.feature_flags.has_package_camera = True
|
||||||
|
@ -33,11 +33,11 @@ async def test_camera_remove(
|
|||||||
|
|
||||||
ufp.api.bootstrap.nvr.system_info.ustorage = None
|
ufp.api.bootstrap.nvr.system_info.ustorage = None
|
||||||
await init_entry(hass, ufp, [doorbell, unadopted_camera])
|
await init_entry(hass, ufp, [doorbell, unadopted_camera])
|
||||||
assert_entity_counts(hass, Platform.EVENT, 1, 1)
|
assert_entity_counts(hass, Platform.EVENT, 3, 3)
|
||||||
await remove_entities(hass, ufp, [doorbell, unadopted_camera])
|
await remove_entities(hass, ufp, [doorbell, unadopted_camera])
|
||||||
assert_entity_counts(hass, Platform.EVENT, 0, 0)
|
assert_entity_counts(hass, Platform.EVENT, 0, 0)
|
||||||
await adopt_devices(hass, ufp, [doorbell, unadopted_camera])
|
await adopt_devices(hass, ufp, [doorbell, unadopted_camera])
|
||||||
assert_entity_counts(hass, Platform.EVENT, 1, 1)
|
assert_entity_counts(hass, Platform.EVENT, 3, 3)
|
||||||
|
|
||||||
|
|
||||||
async def test_doorbell_ring(
|
async def test_doorbell_ring(
|
||||||
@ -50,7 +50,7 @@ async def test_doorbell_ring(
|
|||||||
"""Test a doorbell ring event."""
|
"""Test a doorbell ring event."""
|
||||||
|
|
||||||
await init_entry(hass, ufp, [doorbell, unadopted_camera])
|
await init_entry(hass, ufp, [doorbell, unadopted_camera])
|
||||||
assert_entity_counts(hass, Platform.EVENT, 1, 1)
|
assert_entity_counts(hass, Platform.EVENT, 3, 3)
|
||||||
events: list[HAEvent] = []
|
events: list[HAEvent] = []
|
||||||
|
|
||||||
@callback
|
@callback
|
||||||
@ -152,3 +152,177 @@ async def test_doorbell_ring(
|
|||||||
assert state
|
assert state
|
||||||
assert state.state == timestamp
|
assert state.state == timestamp
|
||||||
unsub()
|
unsub()
|
||||||
|
|
||||||
|
|
||||||
|
async def test_doorbell_nfc_scanned(
|
||||||
|
hass: HomeAssistant,
|
||||||
|
ufp: MockUFPFixture,
|
||||||
|
doorbell: Camera,
|
||||||
|
unadopted_camera: Camera,
|
||||||
|
fixed_now: datetime,
|
||||||
|
) -> None:
|
||||||
|
"""Test a doorbell NFC scanned event."""
|
||||||
|
|
||||||
|
await init_entry(hass, ufp, [doorbell, unadopted_camera])
|
||||||
|
assert_entity_counts(hass, Platform.EVENT, 3, 3)
|
||||||
|
events: list[HAEvent] = []
|
||||||
|
|
||||||
|
@callback
|
||||||
|
def _capture_event(event: HAEvent) -> None:
|
||||||
|
events.append(event)
|
||||||
|
|
||||||
|
_, entity_id = ids_from_device_description(
|
||||||
|
Platform.EVENT, doorbell, EVENT_DESCRIPTIONS[1]
|
||||||
|
)
|
||||||
|
|
||||||
|
unsub = async_track_state_change_event(hass, entity_id, _capture_event)
|
||||||
|
event = Event(
|
||||||
|
model=ModelType.EVENT,
|
||||||
|
id="test_event_id",
|
||||||
|
type=EventType.NFC_CARD_SCANNED,
|
||||||
|
start=fixed_now - timedelta(seconds=1),
|
||||||
|
end=None,
|
||||||
|
score=100,
|
||||||
|
smart_detect_types=[],
|
||||||
|
smart_detect_event_ids=[],
|
||||||
|
camera_id=doorbell.id,
|
||||||
|
api=ufp.api,
|
||||||
|
metadata={"nfc": {"nfc_id": "test_nfc_id", "user_id": "test_user_id"}},
|
||||||
|
)
|
||||||
|
|
||||||
|
new_camera = doorbell.copy()
|
||||||
|
new_camera.last_nfc_card_scanned_event_id = "test_event_id"
|
||||||
|
ufp.api.bootstrap.cameras = {new_camera.id: new_camera}
|
||||||
|
ufp.api.bootstrap.events = {event.id: event}
|
||||||
|
|
||||||
|
mock_msg = Mock()
|
||||||
|
mock_msg.changed_data = {}
|
||||||
|
mock_msg.new_obj = event
|
||||||
|
ufp.ws_msg(mock_msg)
|
||||||
|
|
||||||
|
await hass.async_block_till_done()
|
||||||
|
|
||||||
|
assert len(events) == 1
|
||||||
|
state = events[0].data["new_state"]
|
||||||
|
assert state
|
||||||
|
assert state.attributes[ATTR_ATTRIBUTION] == DEFAULT_ATTRIBUTION
|
||||||
|
assert state.attributes[ATTR_EVENT_ID] == "test_event_id"
|
||||||
|
assert state.attributes["nfc_id"] == "test_nfc_id"
|
||||||
|
|
||||||
|
unsub()
|
||||||
|
|
||||||
|
|
||||||
|
async def test_doorbell_fingerprint_identified(
|
||||||
|
hass: HomeAssistant,
|
||||||
|
ufp: MockUFPFixture,
|
||||||
|
doorbell: Camera,
|
||||||
|
unadopted_camera: Camera,
|
||||||
|
fixed_now: datetime,
|
||||||
|
) -> None:
|
||||||
|
"""Test a doorbell fingerprint identified event."""
|
||||||
|
|
||||||
|
await init_entry(hass, ufp, [doorbell, unadopted_camera])
|
||||||
|
assert_entity_counts(hass, Platform.EVENT, 3, 3)
|
||||||
|
events: list[HAEvent] = []
|
||||||
|
|
||||||
|
@callback
|
||||||
|
def _capture_event(event: HAEvent) -> None:
|
||||||
|
events.append(event)
|
||||||
|
|
||||||
|
_, entity_id = ids_from_device_description(
|
||||||
|
Platform.EVENT, doorbell, EVENT_DESCRIPTIONS[2]
|
||||||
|
)
|
||||||
|
|
||||||
|
unsub = async_track_state_change_event(hass, entity_id, _capture_event)
|
||||||
|
event = Event(
|
||||||
|
model=ModelType.EVENT,
|
||||||
|
id="test_event_id",
|
||||||
|
type=EventType.FINGERPRINT_IDENTIFIED,
|
||||||
|
start=fixed_now - timedelta(seconds=1),
|
||||||
|
end=None,
|
||||||
|
score=100,
|
||||||
|
smart_detect_types=[],
|
||||||
|
smart_detect_event_ids=[],
|
||||||
|
camera_id=doorbell.id,
|
||||||
|
api=ufp.api,
|
||||||
|
metadata={"fingerprint": {"ulp_id": "test_ulp_id"}},
|
||||||
|
)
|
||||||
|
|
||||||
|
new_camera = doorbell.copy()
|
||||||
|
new_camera.last_fingerprint_identified_event_id = "test_event_id"
|
||||||
|
ufp.api.bootstrap.cameras = {new_camera.id: new_camera}
|
||||||
|
ufp.api.bootstrap.events = {event.id: event}
|
||||||
|
|
||||||
|
mock_msg = Mock()
|
||||||
|
mock_msg.changed_data = {}
|
||||||
|
mock_msg.new_obj = event
|
||||||
|
ufp.ws_msg(mock_msg)
|
||||||
|
|
||||||
|
await hass.async_block_till_done()
|
||||||
|
|
||||||
|
assert len(events) == 1
|
||||||
|
state = events[0].data["new_state"]
|
||||||
|
assert state
|
||||||
|
assert state.attributes[ATTR_ATTRIBUTION] == DEFAULT_ATTRIBUTION
|
||||||
|
assert state.attributes[ATTR_EVENT_ID] == "test_event_id"
|
||||||
|
assert state.attributes["ulp_id"] == "test_ulp_id"
|
||||||
|
|
||||||
|
unsub()
|
||||||
|
|
||||||
|
|
||||||
|
async def test_doorbell_fingerprint_not_identified(
|
||||||
|
hass: HomeAssistant,
|
||||||
|
ufp: MockUFPFixture,
|
||||||
|
doorbell: Camera,
|
||||||
|
unadopted_camera: Camera,
|
||||||
|
fixed_now: datetime,
|
||||||
|
) -> None:
|
||||||
|
"""Test a doorbell fingerprint identified event."""
|
||||||
|
|
||||||
|
await init_entry(hass, ufp, [doorbell, unadopted_camera])
|
||||||
|
assert_entity_counts(hass, Platform.EVENT, 3, 3)
|
||||||
|
events: list[HAEvent] = []
|
||||||
|
|
||||||
|
@callback
|
||||||
|
def _capture_event(event: HAEvent) -> None:
|
||||||
|
events.append(event)
|
||||||
|
|
||||||
|
_, entity_id = ids_from_device_description(
|
||||||
|
Platform.EVENT, doorbell, EVENT_DESCRIPTIONS[2]
|
||||||
|
)
|
||||||
|
|
||||||
|
unsub = async_track_state_change_event(hass, entity_id, _capture_event)
|
||||||
|
event = Event(
|
||||||
|
model=ModelType.EVENT,
|
||||||
|
id="test_event_id",
|
||||||
|
type=EventType.FINGERPRINT_IDENTIFIED,
|
||||||
|
start=fixed_now - timedelta(seconds=1),
|
||||||
|
end=None,
|
||||||
|
score=100,
|
||||||
|
smart_detect_types=[],
|
||||||
|
smart_detect_event_ids=[],
|
||||||
|
camera_id=doorbell.id,
|
||||||
|
api=ufp.api,
|
||||||
|
metadata={"fingerprint": {}},
|
||||||
|
)
|
||||||
|
|
||||||
|
new_camera = doorbell.copy()
|
||||||
|
new_camera.last_fingerprint_identified_event_id = "test_event_id"
|
||||||
|
ufp.api.bootstrap.cameras = {new_camera.id: new_camera}
|
||||||
|
ufp.api.bootstrap.events = {event.id: event}
|
||||||
|
|
||||||
|
mock_msg = Mock()
|
||||||
|
mock_msg.changed_data = {}
|
||||||
|
mock_msg.new_obj = event
|
||||||
|
ufp.ws_msg(mock_msg)
|
||||||
|
|
||||||
|
await hass.async_block_till_done()
|
||||||
|
|
||||||
|
assert len(events) == 1
|
||||||
|
state = events[0].data["new_state"]
|
||||||
|
assert state
|
||||||
|
assert state.attributes[ATTR_ATTRIBUTION] == DEFAULT_ATTRIBUTION
|
||||||
|
assert state.attributes[ATTR_EVENT_ID] == "test_event_id"
|
||||||
|
assert state.attributes["ulp_id"] == ""
|
||||||
|
|
||||||
|
unsub()
|
||||||
|
@ -109,7 +109,11 @@ def ids_from_device_description(
|
|||||||
"""Return expected unique_id and entity_id for a give platform/device/description combination."""
|
"""Return expected unique_id and entity_id for a give platform/device/description combination."""
|
||||||
|
|
||||||
entity_name = normalize_name(device.display_name)
|
entity_name = normalize_name(device.display_name)
|
||||||
description_entity_name = normalize_name(str(description.name))
|
|
||||||
|
if description.name and isinstance(description.name, str):
|
||||||
|
description_entity_name = normalize_name(description.name)
|
||||||
|
else:
|
||||||
|
description_entity_name = normalize_name(description.key)
|
||||||
|
|
||||||
unique_id = f"{device.mac}_{description.key}"
|
unique_id = f"{device.mac}_{description.key}"
|
||||||
entity_id = f"{platform.value}.{entity_name}_{description_entity_name}"
|
entity_id = f"{platform.value}.{entity_name}_{description_entity_name}"
|
||||||
|
Loading…
x
Reference in New Issue
Block a user