2022-01-23 21:56:09 +00:00
|
|
|
"""Diagnostics support for HomeKit Controller."""
|
|
|
|
from __future__ import annotations
|
|
|
|
|
|
|
|
from typing import Any
|
|
|
|
|
|
|
|
from aiohomekit.model.characteristics.characteristic_types import CharacteristicsTypes
|
|
|
|
|
|
|
|
from homeassistant.components.diagnostics import REDACTED, async_redact_data
|
|
|
|
from homeassistant.config_entries import ConfigEntry
|
|
|
|
from homeassistant.core import HomeAssistant, callback
|
|
|
|
from homeassistant.helpers import device_registry as dr, entity_registry as er
|
|
|
|
from homeassistant.helpers.device_registry import DeviceEntry
|
|
|
|
|
|
|
|
from .connection import HKDevice
|
|
|
|
from .const import KNOWN_DEVICES
|
|
|
|
|
|
|
|
REDACTED_CHARACTERISTICS = [
|
2022-01-31 22:48:16 +00:00
|
|
|
CharacteristicsTypes.SERIAL_NUMBER,
|
2022-01-23 21:56:09 +00:00
|
|
|
]
|
|
|
|
|
|
|
|
REDACTED_CONFIG_ENTRY_KEYS = [
|
|
|
|
"AccessoryIP",
|
|
|
|
"iOSDeviceLTSK",
|
|
|
|
]
|
|
|
|
|
|
|
|
REDACTED_STATE = ["access_token", "entity_picture"]
|
|
|
|
|
|
|
|
|
|
|
|
async def async_get_config_entry_diagnostics(
|
|
|
|
hass: HomeAssistant, entry: ConfigEntry
|
|
|
|
) -> dict[str, Any]:
|
|
|
|
"""Return diagnostics for a config entry."""
|
|
|
|
return _async_get_diagnostics(hass, entry)
|
|
|
|
|
|
|
|
|
|
|
|
async def async_get_device_diagnostics(
|
|
|
|
hass: HomeAssistant, entry: ConfigEntry, device: DeviceEntry
|
|
|
|
) -> dict[str, Any]:
|
|
|
|
"""Return diagnostics for a device entry."""
|
|
|
|
return _async_get_diagnostics(hass, entry, device)
|
|
|
|
|
|
|
|
|
|
|
|
@callback
|
|
|
|
def _async_get_diagnostics_for_device(
|
|
|
|
hass: HomeAssistant, device: DeviceEntry
|
|
|
|
) -> dict[str, Any]:
|
2022-02-01 11:27:35 +00:00
|
|
|
data: dict[str, Any] = {}
|
2022-01-23 21:56:09 +00:00
|
|
|
|
|
|
|
data["name"] = device.name
|
|
|
|
data["model"] = device.model
|
|
|
|
data["manfacturer"] = device.manufacturer
|
|
|
|
data["sw_version"] = device.sw_version
|
|
|
|
data["hw_version"] = device.hw_version
|
|
|
|
|
|
|
|
entities = data["entities"] = []
|
|
|
|
|
|
|
|
hass_entities = er.async_entries_for_device(
|
|
|
|
er.async_get(hass),
|
|
|
|
device_id=device.id,
|
|
|
|
include_disabled_entities=True,
|
|
|
|
)
|
|
|
|
|
2022-02-01 11:27:35 +00:00
|
|
|
hass_entities.sort(key=lambda entry: entry.original_name or "")
|
2022-01-24 08:42:55 +00:00
|
|
|
|
2022-01-23 21:56:09 +00:00
|
|
|
for entity_entry in hass_entities:
|
|
|
|
state = hass.states.get(entity_entry.entity_id)
|
|
|
|
state_dict = None
|
|
|
|
if state:
|
|
|
|
state_dict = async_redact_data(state.as_dict(), REDACTED_STATE)
|
|
|
|
state_dict.pop("context", None)
|
|
|
|
|
|
|
|
entities.append(
|
|
|
|
{
|
|
|
|
"original_name": entity_entry.original_name,
|
|
|
|
"original_device_class": entity_entry.original_device_class,
|
|
|
|
"entity_category": entity_entry.entity_category,
|
|
|
|
"original_icon": entity_entry.original_icon,
|
|
|
|
"icon": entity_entry.icon,
|
|
|
|
"unit_of_measurement": entity_entry.unit_of_measurement,
|
|
|
|
"device_class": entity_entry.device_class,
|
|
|
|
"disabled": entity_entry.disabled,
|
|
|
|
"disabled_by": entity_entry.disabled_by,
|
|
|
|
"state": state_dict,
|
|
|
|
}
|
|
|
|
)
|
|
|
|
|
|
|
|
return data
|
|
|
|
|
|
|
|
|
|
|
|
@callback
|
|
|
|
def _async_get_diagnostics(
|
|
|
|
hass: HomeAssistant, entry: ConfigEntry, device: DeviceEntry | None = None
|
|
|
|
) -> dict[str, Any]:
|
|
|
|
"""Return diagnostics for a config entry."""
|
|
|
|
hkid = entry.data["AccessoryPairingID"]
|
|
|
|
connection: HKDevice = hass.data[KNOWN_DEVICES][hkid]
|
|
|
|
|
2022-02-01 11:27:35 +00:00
|
|
|
data: dict[str, Any] = {
|
2022-01-23 21:56:09 +00:00
|
|
|
"config-entry": {
|
|
|
|
"title": entry.title,
|
|
|
|
"version": entry.version,
|
|
|
|
"data": async_redact_data(entry.data, REDACTED_CONFIG_ENTRY_KEYS),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
# This is the raw data as returned by homekit
|
|
|
|
# It is roughly equivalent to what is in .storage/homekit_controller-entity-map
|
|
|
|
# But it also has the latest values seen by the polling or events
|
|
|
|
data["entity-map"] = accessories = connection.entity_map.serialize()
|
2022-07-16 17:49:15 +00:00
|
|
|
data["config-num"] = connection.config_num
|
2022-01-23 21:56:09 +00:00
|
|
|
|
|
|
|
# It contains serial numbers, which we should strip out
|
|
|
|
for accessory in accessories:
|
|
|
|
for service in accessory.get("services", []):
|
|
|
|
for char in service.get("characteristics", []):
|
2022-01-31 22:48:16 +00:00
|
|
|
if char["type"] in REDACTED_CHARACTERISTICS:
|
2022-01-23 21:56:09 +00:00
|
|
|
char["value"] = REDACTED
|
|
|
|
|
|
|
|
if device:
|
|
|
|
data["device"] = _async_get_diagnostics_for_device(hass, device)
|
|
|
|
else:
|
|
|
|
device_registry = dr.async_get(hass)
|
|
|
|
|
|
|
|
devices = data["devices"] = []
|
|
|
|
for device_id in connection.devices.values():
|
2022-02-19 16:21:26 +00:00
|
|
|
if not (device := device_registry.async_get(device_id)):
|
2022-02-01 11:27:35 +00:00
|
|
|
continue
|
2022-01-23 21:56:09 +00:00
|
|
|
devices.append(_async_get_diagnostics_for_device(hass, device))
|
|
|
|
|
|
|
|
return data
|