mirror of
https://github.com/home-assistant/core.git
synced 2026-05-15 13:01:22 +01:00
3f7070fd54
Co-authored-by: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
138 lines
4.9 KiB
Python
138 lines
4.9 KiB
Python
"""Test the switchbot event entities."""
|
|
|
|
from collections.abc import Callable
|
|
from datetime import UTC, datetime, timedelta
|
|
from itertools import count
|
|
from unittest.mock import patch
|
|
|
|
import pytest
|
|
|
|
from homeassistant.components.bluetooth import BluetoothServiceInfoBleak
|
|
from homeassistant.components.switchbot.const import DOMAIN
|
|
from homeassistant.const import STATE_UNKNOWN
|
|
from homeassistant.core import Event, EventStateChangedData, HomeAssistant, callback
|
|
from homeassistant.helpers.event import async_track_state_change_event
|
|
from homeassistant.setup import async_setup_component
|
|
|
|
from . import KEYPAD_VISION_PRO_INFO
|
|
|
|
from tests.common import MockConfigEntry
|
|
from tests.components.bluetooth import (
|
|
generate_advertisement_data,
|
|
generate_ble_device,
|
|
inject_bluetooth_service_info,
|
|
)
|
|
|
|
|
|
def _with_doorbell_seq(
|
|
info: BluetoothServiceInfoBleak, seq: int = 1
|
|
) -> BluetoothServiceInfoBleak:
|
|
"""Return a BLE service info with the doorbell seq bits set."""
|
|
mfr_data = bytearray(info.manufacturer_data[2409])
|
|
mfr_data[12] = (mfr_data[12] & 0b11111000) | (seq & 0b00000111)
|
|
updated_mfr_data = {2409: bytes(mfr_data)}
|
|
return BluetoothServiceInfoBleak(
|
|
name=info.name,
|
|
manufacturer_data=updated_mfr_data,
|
|
service_data=info.service_data,
|
|
service_uuids=info.service_uuids,
|
|
address=info.address,
|
|
rssi=info.rssi,
|
|
source=info.source,
|
|
advertisement=generate_advertisement_data(
|
|
local_name=info.name,
|
|
manufacturer_data=updated_mfr_data,
|
|
service_data=info.service_data,
|
|
service_uuids=info.service_uuids,
|
|
),
|
|
device=generate_ble_device(info.address, info.name),
|
|
time=info.time,
|
|
connectable=info.connectable,
|
|
tx_power=info.tx_power,
|
|
)
|
|
|
|
|
|
@pytest.mark.usefixtures("entity_registry_enabled_by_default")
|
|
async def test_keypad_vision_pro_doorbell_event(
|
|
hass: HomeAssistant,
|
|
mock_entry_encrypted_factory: Callable[[str], MockConfigEntry],
|
|
) -> None:
|
|
"""Test keypad vision pro doorbell event uses doorbell_seq for detection."""
|
|
# Make each ring's timestamp distinct so the entity's state value (which is
|
|
# the iso-formatted timestamp at millisecond resolution) differs between
|
|
# rings, ensuring state_changed fires for each. We avoid the `freezer`
|
|
# fixture here because it patches time.monotonic, which warps loop.time
|
|
# forward and causes the bluetooth manager's pre-scheduled unavailability
|
|
# check to fire immediately and mark the injected device unavailable.
|
|
timestamps = (
|
|
datetime(2026, 1, 1, tzinfo=UTC) + timedelta(seconds=i) for i in count()
|
|
)
|
|
|
|
await async_setup_component(hass, DOMAIN, {})
|
|
inject_bluetooth_service_info(hass, KEYPAD_VISION_PRO_INFO)
|
|
|
|
entry = mock_entry_encrypted_factory(sensor_type="keypad_vision_pro")
|
|
entry.add_to_hass(hass)
|
|
|
|
entity_id = "event.test_name_doorbell"
|
|
ring_states: list[str] = []
|
|
|
|
@callback
|
|
def _track_ring(event: Event[EventStateChangedData]) -> None:
|
|
new_state = event.data["new_state"]
|
|
if new_state and new_state.attributes.get("event_type") == "ring":
|
|
ring_states.append(new_state.state)
|
|
|
|
with (
|
|
patch(
|
|
"homeassistant.components.switchbot.sensor.switchbot.SwitchbotKeypadVision.update",
|
|
return_value=True,
|
|
),
|
|
patch(
|
|
"homeassistant.components.event.dt_util.utcnow",
|
|
side_effect=lambda: next(timestamps),
|
|
),
|
|
):
|
|
assert await hass.config_entries.async_setup(entry.entry_id)
|
|
await hass.async_block_till_done()
|
|
|
|
async_track_state_change_event(hass, entity_id, _track_ring)
|
|
|
|
state = hass.states.get(entity_id)
|
|
assert state
|
|
assert state.state == STATE_UNKNOWN
|
|
|
|
# First ring: seq changes from 0 → 1
|
|
inject_bluetooth_service_info(
|
|
hass, _with_doorbell_seq(KEYPAD_VISION_PRO_INFO, 1)
|
|
)
|
|
await hass.async_block_till_done()
|
|
assert len(ring_states) == 1
|
|
|
|
# Same seq repeated — no new ring event
|
|
inject_bluetooth_service_info(
|
|
hass, _with_doorbell_seq(KEYPAD_VISION_PRO_INFO, 1)
|
|
)
|
|
await hass.async_block_till_done()
|
|
assert len(ring_states) == 1
|
|
|
|
# Second ring: seq changes from 1 → 2
|
|
inject_bluetooth_service_info(
|
|
hass, _with_doorbell_seq(KEYPAD_VISION_PRO_INFO, 2)
|
|
)
|
|
await hass.async_block_till_done()
|
|
assert len(ring_states) == 2
|
|
|
|
# Seq wraps from 7 → 1 — still a ring
|
|
inject_bluetooth_service_info(
|
|
hass, _with_doorbell_seq(KEYPAD_VISION_PRO_INFO, 7)
|
|
)
|
|
await hass.async_block_till_done()
|
|
assert len(ring_states) == 3
|
|
|
|
inject_bluetooth_service_info(
|
|
hass, _with_doorbell_seq(KEYPAD_VISION_PRO_INFO, 1)
|
|
)
|
|
await hass.async_block_till_done()
|
|
assert len(ring_states) == 4
|