mirror of
https://github.com/home-assistant/core.git
synced 2026-02-14 23:28:42 +00:00
Portainer fix multiple environments & containers (#153674)
This commit is contained in:
@@ -2,6 +2,8 @@
|
||||
|
||||
from __future__ import annotations
|
||||
|
||||
import logging
|
||||
|
||||
from pyportainer import Portainer
|
||||
|
||||
from homeassistant.config_entries import ConfigEntry
|
||||
@@ -16,6 +18,8 @@ from homeassistant.const import (
|
||||
from homeassistant.core import HomeAssistant
|
||||
from homeassistant.helpers.aiohttp_client import async_create_clientsession
|
||||
import homeassistant.helpers.config_validation as cv
|
||||
import homeassistant.helpers.device_registry as dr
|
||||
import homeassistant.helpers.entity_registry as er
|
||||
from homeassistant.helpers.typing import ConfigType
|
||||
|
||||
from .const import DOMAIN
|
||||
@@ -33,6 +37,8 @@ CONFIG_SCHEMA = cv.config_entry_only_config_schema(DOMAIN)
|
||||
|
||||
type PortainerConfigEntry = ConfigEntry[PortainerCoordinator]
|
||||
|
||||
_LOGGER = logging.getLogger(__name__)
|
||||
|
||||
|
||||
async def async_setup_entry(hass: HomeAssistant, entry: PortainerConfigEntry) -> bool:
|
||||
"""Set up Portainer from a config entry."""
|
||||
@@ -79,4 +85,55 @@ async def async_migrate_entry(hass: HomeAssistant, entry: PortainerConfigEntry)
|
||||
data[CONF_VERIFY_SSL] = True
|
||||
hass.config_entries.async_update_entry(entry=entry, data=data, version=3)
|
||||
|
||||
if entry.version < 4:
|
||||
device_registry = dr.async_get(hass)
|
||||
entity_registry = er.async_get(hass)
|
||||
devices = dr.async_entries_for_config_entry(device_registry, entry.entry_id)
|
||||
for device in devices:
|
||||
# This means it's an endpoint. This can be skipped, we're only interested in the containers
|
||||
if device.via_device_id is None:
|
||||
continue
|
||||
|
||||
parent_devices = device_registry.async_get(device.via_device_id)
|
||||
assert parent_devices
|
||||
for parent_device in parent_devices.identifiers:
|
||||
if parent_device[0] == DOMAIN:
|
||||
parent_device_identifiers = parent_device
|
||||
break
|
||||
|
||||
_LOGGER.debug("Parent device identifiers: %s", parent_device_identifiers)
|
||||
endpoint_id = parent_device_identifiers[1].split("_")[-1]
|
||||
_LOGGER.debug("Endpoint ID: %s", endpoint_id)
|
||||
current_identifier = next(iter(device.identifiers))
|
||||
_LOGGER.debug("Current identifier: %s", current_identifier)
|
||||
container = current_identifier[1].split("_", 1)[1]
|
||||
_LOGGER.debug("Container name: %s", container)
|
||||
new_identifier = f"{entry.entry_id}_{endpoint_id}_{container}"
|
||||
_LOGGER.debug("New identifier: %s", new_identifier)
|
||||
|
||||
new_identifiers = set(device.identifiers)
|
||||
new_identifiers.add((DOMAIN, new_identifier))
|
||||
|
||||
device_registry.async_update_device(
|
||||
device.id,
|
||||
new_identifiers=new_identifiers,
|
||||
)
|
||||
|
||||
# Now also update the underlying entities with the new unique_attr_id
|
||||
entities_device = er.async_entries_for_device(
|
||||
entity_registry,
|
||||
device.id,
|
||||
)
|
||||
for entity in entities_device:
|
||||
_LOGGER.debug("Handling entity: %s", entity)
|
||||
# This time we also also have a rest tail (for instance _firefly_iii_db)
|
||||
_, _, rest_tail = entity.unique_id.split("_", 2)
|
||||
new_unique_id = f"{new_identifier}_{rest_tail}"
|
||||
_LOGGER.debug("New unique ID: %s", new_unique_id)
|
||||
entity_registry.async_update_entity(
|
||||
entity_id=entity.entity_id, new_unique_id=new_unique_id
|
||||
)
|
||||
|
||||
hass.config_entries.async_update_entry(entry=entry, version=4)
|
||||
|
||||
return True
|
||||
|
||||
@@ -55,7 +55,7 @@ async def _validate_input(hass: HomeAssistant, data: dict[str, Any]) -> None:
|
||||
class PortainerConfigFlow(ConfigFlow, domain=DOMAIN):
|
||||
"""Handle a config flow for Portainer."""
|
||||
|
||||
VERSION = 2
|
||||
VERSION = 4
|
||||
|
||||
async def async_step_user(
|
||||
self, user_input: dict[str, Any] | None = None
|
||||
|
||||
@@ -75,7 +75,10 @@ class PortainerContainerEntity(PortainerCoordinatorEntity):
|
||||
|
||||
self._attr_device_info = DeviceInfo(
|
||||
identifiers={
|
||||
(DOMAIN, f"{self.coordinator.config_entry.entry_id}_{self.device_name}")
|
||||
(
|
||||
DOMAIN,
|
||||
f"{self.coordinator.config_entry.entry_id}_{self.endpoint_id}_{self.device_name}",
|
||||
)
|
||||
},
|
||||
manufacturer=DEFAULT_NAME,
|
||||
configuration_url=URL(
|
||||
|
||||
@@ -22,7 +22,7 @@
|
||||
]),
|
||||
'title': 'Portainer test',
|
||||
'unique_id': 'test_api_token',
|
||||
'version': 2,
|
||||
'version': 4,
|
||||
}),
|
||||
'coordinator': dict({
|
||||
'endpoints': list([
|
||||
|
||||
@@ -19,6 +19,7 @@ from homeassistant.const import (
|
||||
CONF_VERIFY_SSL,
|
||||
)
|
||||
from homeassistant.core import HomeAssistant
|
||||
from homeassistant.helpers import device_registry as dr, entity_registry as er
|
||||
|
||||
from . import setup_integration
|
||||
|
||||
@@ -63,9 +64,72 @@ async def test_migrations(hass: HomeAssistant) -> None:
|
||||
await hass.config_entries.async_setup(entry.entry_id)
|
||||
await hass.async_block_till_done()
|
||||
|
||||
assert entry.version == 3
|
||||
assert CONF_HOST not in entry.data
|
||||
assert CONF_API_KEY not in entry.data
|
||||
assert entry.data[CONF_URL] == "http://test_host"
|
||||
assert entry.data[CONF_API_TOKEN] == "test_key"
|
||||
assert entry.data[CONF_VERIFY_SSL] is True
|
||||
|
||||
# Confirm we went through all current migrations
|
||||
assert entry.version == 4
|
||||
|
||||
|
||||
async def test_migration_v3_to_v4(
|
||||
hass: HomeAssistant,
|
||||
device_registry: dr.DeviceRegistry,
|
||||
entity_registry: er.EntityRegistry,
|
||||
) -> None:
|
||||
"""Test migration from v3 config entry."""
|
||||
entry = MockConfigEntry(
|
||||
domain=DOMAIN,
|
||||
data={
|
||||
CONF_HOST: "http://test_host",
|
||||
CONF_API_KEY: "test_key",
|
||||
},
|
||||
unique_id="1",
|
||||
version=3,
|
||||
)
|
||||
entry.add_to_hass(hass)
|
||||
assert entry.version == 3
|
||||
|
||||
endpoint_device = device_registry.async_get_or_create(
|
||||
config_entry_id=entry.entry_id,
|
||||
identifiers={(DOMAIN, f"{entry.entry_id}_endpoint_1")},
|
||||
name="Test Endpoint",
|
||||
)
|
||||
|
||||
original_container_identifier = f"{entry.entry_id}_adguard"
|
||||
container_device = device_registry.async_get_or_create(
|
||||
config_entry_id=entry.entry_id,
|
||||
identifiers={(DOMAIN, original_container_identifier)},
|
||||
via_device=(DOMAIN, f"{entry.entry_id}_endpoint_1"),
|
||||
name="Test Container",
|
||||
)
|
||||
|
||||
container_entity = entity_registry.async_get_or_create(
|
||||
domain="switch",
|
||||
platform=DOMAIN,
|
||||
unique_id=f"{entry.entry_id}_adguard_container",
|
||||
config_entry=entry,
|
||||
device_id=container_device.id,
|
||||
original_name="Test Container Switch",
|
||||
)
|
||||
|
||||
assert container_device.via_device_id == endpoint_device.id
|
||||
assert container_device.identifiers == {(DOMAIN, original_container_identifier)}
|
||||
assert container_entity.unique_id == f"{entry.entry_id}_adguard_container"
|
||||
|
||||
await hass.config_entries.async_setup(entry.entry_id)
|
||||
await hass.async_block_till_done()
|
||||
|
||||
assert entry.version == 4
|
||||
|
||||
# Fetch again, to assert the new identifiers
|
||||
container_after = device_registry.async_get(container_device.id)
|
||||
entity_after = entity_registry.async_get(container_entity.entity_id)
|
||||
|
||||
assert container_after.identifiers == {
|
||||
(DOMAIN, original_container_identifier),
|
||||
(DOMAIN, f"{entry.entry_id}_1_adguard"),
|
||||
}
|
||||
assert entity_after.unique_id == f"{entry.entry_id}_1_adguard_container"
|
||||
|
||||
Reference in New Issue
Block a user