handle device_id being changed

This commit is contained in:
J. Nick Koston
2025-06-24 19:01:45 +02:00
parent 5c09591f54
commit a5d4d636bc
2 changed files with 140 additions and 25 deletions

View File

@ -33,12 +33,7 @@ from homeassistant.helpers.entity_platform import AddEntitiesCallback
from .const import DOMAIN
# Import config flow so that it's added to the registry
from .entry_data import (
ESPHomeConfigEntry,
RuntimeEntryData,
build_device_unique_id,
build_unique_id,
)
from .entry_data import ESPHomeConfigEntry, RuntimeEntryData, build_device_unique_id
from .enum_mapper import EsphomeEnumMapper
_LOGGER = logging.getLogger(__name__)
@ -85,30 +80,24 @@ def async_static_info_updated(
# Entity has switched devices, need to migrate unique_id
old_unique_id = build_device_unique_id(device_info.mac_address, old_info)
new_unique_id = build_device_unique_id(device_info.mac_address, info)
entity_id = ent_reg.async_get_entity_id(platform.domain, DOMAIN, old_unique_id)
# If not found by exact match, search for entity with base unique_id
# This happens when the YAML config was modified and renamed the device_id
# If entity not found in registry, re-add it
# This happens when the device_id changed and the old device was deleted
if entity_id is None:
base_unique_id = build_unique_id(device_info.mac_address, info)
# Search all entities for this config entry
for entry in ent_reg.entities.get_entries_for_config_entry_id(
entry_data.entry_id
):
if entry.platform != DOMAIN or entry.domain != platform.domain:
continue
# Check if it's the exact base unique_id or starts with base_unique_id@
if entry.unique_id == base_unique_id or (
entry.unique_id and entry.unique_id.startswith(f"{base_unique_id}@")
):
entity_id = entry.entity_id
break
# Entity must exist in registry since we found it in current_infos
assert entity_id is not None
_LOGGER.info(
"Entity with old unique_id %s not found in registry after device_id "
"changed from %s to %s, re-adding entity",
old_unique_id,
old_info.device_id,
info.device_id,
)
entity = entity_type(entry_data, platform.domain, info, state_type)
add_entities.append(entity)
continue
updates: dict[str, Any] = {}
new_unique_id = build_device_unique_id(device_info.mac_address, info)
# Update unique_id if it changed
if old_unique_id != new_unique_id:

View File

@ -1471,3 +1471,129 @@ async def test_unique_id_migration_between_sub_devices(
)
assert bedroom_device is not None
assert entity_entry.device_id == bedroom_device.id
async def test_entity_device_id_rename_in_yaml(
hass: HomeAssistant,
entity_registry: er.EntityRegistry,
device_registry: dr.DeviceRegistry,
mock_client: APIClient,
mock_esphome_device: MockESPHomeDeviceType,
) -> None:
"""Test that entities are re-added as new when user renames device_id in YAML config."""
# Initial setup: entity on sub-device with device_id 11111111
sub_devices = [
SubDeviceInfo(device_id=11111111, name="old_device", area_id=0),
]
device_info = {
"name": "test",
"devices": sub_devices,
}
# Entity on sub-device
entity_info = [
BinarySensorInfo(
object_id="sensor",
key=1,
name="Sensor",
unique_id="unused",
device_id=11111111,
),
]
states = [
BinarySensorState(key=1, state=True, missing_state=False),
]
device = await mock_esphome_device(
mock_client=mock_client,
device_info=device_info,
entity_info=entity_info,
states=states,
)
# Verify initial entity setup
state = hass.states.get("binary_sensor.old_device_sensor")
assert state is not None
assert state.state == STATE_ON
# Wait for entity to be registered
await hass.async_block_till_done()
# Get the entity from registry
entity_entry = entity_registry.async_get("binary_sensor.old_device_sensor")
assert entity_entry is not None
initial_unique_id = entity_entry.unique_id
# Should have @11111111 suffix
assert "@11111111" in initial_unique_id
# Simulate user renaming device_id in YAML config
# The device_id hash changes from 11111111 to 99999999
# This is treated as a completely new device
renamed_sub_devices = [
SubDeviceInfo(device_id=99999999, name="renamed_device", area_id=0),
]
# Get the config entry from hass
entries = hass.config_entries.async_entries(DOMAIN)
assert len(entries) == 1
entry = entries[0]
# Update device_id_to_name mapping
entry_data = entry.runtime_data
entry_data.device_id_to_name = {
sub_device.device_id: sub_device.name for sub_device in renamed_sub_devices
}
# Create new DeviceInfo with renamed device
current_device_info = mock_client.device_info.return_value
device_info_dict = asdict(current_device_info)
device_info_dict["devices"] = renamed_sub_devices
new_device_info = DeviceInfo(**device_info_dict)
mock_client.device_info.return_value = new_device_info
# Entity info now has the new device_id
new_entity_info = [
BinarySensorInfo(
object_id="sensor", # Same object_id
key=1, # Same key
name="Sensor",
unique_id="unused",
device_id=99999999, # New device_id after rename
),
]
# Update the entity info
mock_client.list_entities_services = AsyncMock(return_value=(new_entity_info, []))
# Trigger a reconnect to simulate the YAML config change
await device.mock_disconnect(expected_disconnect=False)
await device.mock_connect()
await hass.async_block_till_done()
# The old entity should be gone (device was deleted)
state = hass.states.get("binary_sensor.old_device_sensor")
assert state is None
# A new entity should exist with a new entity_id based on the new device name
# This is a completely new entity, not a migrated one
state = hass.states.get("binary_sensor.renamed_device_sensor")
assert state is not None
assert state.state == STATE_ON
# Get the new entity from registry
entity_entry = entity_registry.async_get("binary_sensor.renamed_device_sensor")
assert entity_entry is not None
# Unique ID should have the new device_id
base_unique_id = initial_unique_id.replace("@11111111", "")
expected_unique_id = f"{base_unique_id}@99999999"
assert entity_entry.unique_id == expected_unique_id
# Entity should be associated with the new device
renamed_device = device_registry.async_get_device(
identifiers={(DOMAIN, f"{device.device_info.mac_address}_99999999")}
)
assert renamed_device is not None
assert entity_entry.device_id == renamed_device.id