2019-09-14 13:15:06 +00:00
|
|
|
"""deCONZ services."""
|
2020-12-02 15:21:27 +00:00
|
|
|
|
2021-11-14 18:47:15 +00:00
|
|
|
from types import MappingProxyType
|
|
|
|
|
2020-02-01 17:11:05 +00:00
|
|
|
from pydeconz.utils import normalize_bridge_id
|
2019-09-14 13:15:06 +00:00
|
|
|
import voluptuous as vol
|
|
|
|
|
2021-11-14 18:47:15 +00:00
|
|
|
from homeassistant.core import HomeAssistant, ServiceCall, callback
|
2021-10-20 09:16:28 +00:00
|
|
|
from homeassistant.helpers import (
|
|
|
|
config_validation as cv,
|
|
|
|
device_registry as dr,
|
|
|
|
entity_registry as er,
|
|
|
|
)
|
2020-10-17 16:44:23 +00:00
|
|
|
from homeassistant.helpers.device_registry import CONNECTION_NETWORK_MAC
|
2020-09-30 18:11:41 +00:00
|
|
|
from homeassistant.helpers.entity_registry import (
|
|
|
|
async_entries_for_config_entry,
|
|
|
|
async_entries_for_device,
|
|
|
|
)
|
2019-09-14 13:15:06 +00:00
|
|
|
|
|
|
|
from .config_flow import get_master_gateway
|
2021-10-07 10:48:27 +00:00
|
|
|
from .const import CONF_BRIDGE_ID, DOMAIN, LOGGER
|
2021-12-13 19:03:01 +00:00
|
|
|
from .gateway import DeconzGateway
|
2019-09-14 13:15:06 +00:00
|
|
|
|
|
|
|
DECONZ_SERVICES = "deconz_services"
|
|
|
|
|
|
|
|
SERVICE_FIELD = "field"
|
|
|
|
SERVICE_ENTITY = "entity"
|
|
|
|
SERVICE_DATA = "data"
|
|
|
|
|
|
|
|
SERVICE_CONFIGURE_DEVICE = "configure"
|
|
|
|
SERVICE_CONFIGURE_DEVICE_SCHEMA = vol.All(
|
|
|
|
vol.Schema(
|
|
|
|
{
|
|
|
|
vol.Optional(SERVICE_ENTITY): cv.entity_id,
|
|
|
|
vol.Optional(SERVICE_FIELD): cv.matches_regex("/.*"),
|
|
|
|
vol.Required(SERVICE_DATA): dict,
|
2020-01-17 23:28:34 +00:00
|
|
|
vol.Optional(CONF_BRIDGE_ID): str,
|
2019-09-14 13:15:06 +00:00
|
|
|
}
|
|
|
|
),
|
|
|
|
cv.has_at_least_one_key(SERVICE_ENTITY, SERVICE_FIELD),
|
|
|
|
)
|
|
|
|
|
|
|
|
SERVICE_DEVICE_REFRESH = "device_refresh"
|
2020-09-30 18:11:41 +00:00
|
|
|
SERVICE_REMOVE_ORPHANED_ENTRIES = "remove_orphaned_entries"
|
|
|
|
SELECT_GATEWAY_SCHEMA = vol.All(vol.Schema({vol.Optional(CONF_BRIDGE_ID): str}))
|
2019-09-14 13:15:06 +00:00
|
|
|
|
2021-10-01 18:31:38 +00:00
|
|
|
SUPPORTED_SERVICES = (
|
|
|
|
SERVICE_CONFIGURE_DEVICE,
|
|
|
|
SERVICE_DEVICE_REFRESH,
|
|
|
|
SERVICE_REMOVE_ORPHANED_ENTRIES,
|
|
|
|
)
|
|
|
|
|
|
|
|
SERVICE_TO_SCHEMA = {
|
|
|
|
SERVICE_CONFIGURE_DEVICE: SERVICE_CONFIGURE_DEVICE_SCHEMA,
|
|
|
|
SERVICE_DEVICE_REFRESH: SELECT_GATEWAY_SCHEMA,
|
|
|
|
SERVICE_REMOVE_ORPHANED_ENTRIES: SELECT_GATEWAY_SCHEMA,
|
|
|
|
}
|
2019-09-14 13:15:06 +00:00
|
|
|
|
|
|
|
|
2021-10-01 18:31:38 +00:00
|
|
|
@callback
|
2021-11-14 18:47:15 +00:00
|
|
|
def async_setup_services(hass: HomeAssistant) -> None:
|
2021-10-01 18:31:38 +00:00
|
|
|
"""Set up services for deCONZ integration."""
|
2019-09-14 13:15:06 +00:00
|
|
|
|
2021-11-14 18:47:15 +00:00
|
|
|
async def async_call_deconz_service(service_call: ServiceCall) -> None:
|
2019-09-14 13:15:06 +00:00
|
|
|
"""Call correct deCONZ service."""
|
|
|
|
service = service_call.service
|
|
|
|
service_data = service_call.data
|
|
|
|
|
2021-07-06 15:18:54 +00:00
|
|
|
if CONF_BRIDGE_ID in service_data:
|
|
|
|
found_gateway = False
|
|
|
|
bridge_id = normalize_bridge_id(service_data[CONF_BRIDGE_ID])
|
|
|
|
|
|
|
|
for possible_gateway in hass.data[DOMAIN].values():
|
|
|
|
if possible_gateway.bridgeid == bridge_id:
|
|
|
|
gateway = possible_gateway
|
|
|
|
found_gateway = True
|
|
|
|
break
|
|
|
|
|
|
|
|
if not found_gateway:
|
|
|
|
LOGGER.error("Could not find the gateway %s", bridge_id)
|
|
|
|
return
|
2021-12-01 17:59:52 +00:00
|
|
|
else:
|
|
|
|
try:
|
|
|
|
gateway = get_master_gateway(hass)
|
|
|
|
except ValueError:
|
|
|
|
LOGGER.error("No master gateway available")
|
|
|
|
return
|
2021-07-06 15:18:54 +00:00
|
|
|
|
2019-09-14 13:15:06 +00:00
|
|
|
if service == SERVICE_CONFIGURE_DEVICE:
|
2021-07-06 15:18:54 +00:00
|
|
|
await async_configure_service(gateway, service_data)
|
2019-09-14 13:15:06 +00:00
|
|
|
|
|
|
|
elif service == SERVICE_DEVICE_REFRESH:
|
2021-07-06 15:18:54 +00:00
|
|
|
await async_refresh_devices_service(gateway)
|
2019-09-14 13:15:06 +00:00
|
|
|
|
2020-09-30 18:11:41 +00:00
|
|
|
elif service == SERVICE_REMOVE_ORPHANED_ENTRIES:
|
2021-07-06 15:18:54 +00:00
|
|
|
await async_remove_orphaned_entries_service(gateway)
|
2020-09-30 18:11:41 +00:00
|
|
|
|
2021-10-01 18:31:38 +00:00
|
|
|
for service in SUPPORTED_SERVICES:
|
|
|
|
hass.services.async_register(
|
|
|
|
DOMAIN,
|
|
|
|
service,
|
|
|
|
async_call_deconz_service,
|
|
|
|
schema=SERVICE_TO_SCHEMA[service],
|
|
|
|
)
|
2019-09-14 13:15:06 +00:00
|
|
|
|
|
|
|
|
2021-10-01 18:31:38 +00:00
|
|
|
@callback
|
2021-11-14 18:47:15 +00:00
|
|
|
def async_unload_services(hass: HomeAssistant) -> None:
|
2021-10-01 18:31:38 +00:00
|
|
|
"""Unload deCONZ services."""
|
|
|
|
for service in SUPPORTED_SERVICES:
|
|
|
|
hass.services.async_remove(DOMAIN, service)
|
2019-09-14 13:15:06 +00:00
|
|
|
|
|
|
|
|
2021-11-14 18:47:15 +00:00
|
|
|
async def async_configure_service(
|
|
|
|
gateway: DeconzGateway, data: MappingProxyType
|
|
|
|
) -> None:
|
2019-09-14 13:15:06 +00:00
|
|
|
"""Set attribute of device in deCONZ.
|
|
|
|
|
|
|
|
Entity is used to resolve to a device path (e.g. '/lights/1').
|
|
|
|
Field is a string representing either a full path
|
|
|
|
(e.g. '/lights/1/state') when entity is not specified, or a
|
|
|
|
subpath (e.g. '/state') when used together with entity.
|
|
|
|
Data is a json object with what data you want to alter
|
|
|
|
e.g. data={'on': true}.
|
|
|
|
{
|
|
|
|
"field": "/lights/1/state",
|
|
|
|
"data": {"on": true}
|
|
|
|
}
|
|
|
|
See Dresden Elektroniks REST API documentation for details:
|
|
|
|
http://dresden-elektronik.github.io/deconz-rest-doc/rest/
|
|
|
|
"""
|
|
|
|
field = data.get(SERVICE_FIELD, "")
|
|
|
|
entity_id = data.get(SERVICE_ENTITY)
|
|
|
|
data = data[SERVICE_DATA]
|
|
|
|
|
|
|
|
if entity_id:
|
|
|
|
try:
|
|
|
|
field = gateway.deconz_ids[entity_id] + field
|
|
|
|
except KeyError:
|
2020-02-01 19:02:57 +00:00
|
|
|
LOGGER.error("Could not find the entity %s", entity_id)
|
2019-09-14 13:15:06 +00:00
|
|
|
return
|
|
|
|
|
2019-12-08 15:53:34 +00:00
|
|
|
await gateway.api.request("put", field, json=data)
|
2019-09-14 13:15:06 +00:00
|
|
|
|
|
|
|
|
2021-11-14 18:47:15 +00:00
|
|
|
async def async_refresh_devices_service(gateway: DeconzGateway) -> None:
|
2019-09-14 13:15:06 +00:00
|
|
|
"""Refresh available devices from deCONZ."""
|
2020-09-30 15:24:30 +00:00
|
|
|
gateway.ignore_state_updates = True
|
|
|
|
await gateway.api.refresh_state()
|
|
|
|
gateway.ignore_state_updates = False
|
2019-09-14 13:15:06 +00:00
|
|
|
|
2021-10-07 10:48:27 +00:00
|
|
|
for resource_type in gateway.deconz_resource_type_to_signal_new_device:
|
|
|
|
gateway.async_add_device_callback(resource_type, force=True)
|
2020-09-30 18:11:41 +00:00
|
|
|
|
|
|
|
|
2021-11-14 18:47:15 +00:00
|
|
|
async def async_remove_orphaned_entries_service(gateway: DeconzGateway) -> None:
|
2020-09-30 18:11:41 +00:00
|
|
|
"""Remove orphaned deCONZ entries from device and entity registries."""
|
2021-10-20 09:16:28 +00:00
|
|
|
device_registry = dr.async_get(gateway.hass)
|
|
|
|
entity_registry = er.async_get(gateway.hass)
|
2020-09-30 18:11:41 +00:00
|
|
|
|
|
|
|
entity_entries = async_entries_for_config_entry(
|
|
|
|
entity_registry, gateway.config_entry.entry_id
|
|
|
|
)
|
|
|
|
|
|
|
|
entities_to_be_removed = []
|
|
|
|
devices_to_be_removed = [
|
|
|
|
entry.id
|
|
|
|
for entry in device_registry.devices.values()
|
|
|
|
if gateway.config_entry.entry_id in entry.config_entries
|
|
|
|
]
|
|
|
|
|
2020-10-17 16:44:23 +00:00
|
|
|
# Don't remove the Gateway host entry
|
|
|
|
gateway_host = device_registry.async_get_device(
|
|
|
|
connections={(CONNECTION_NETWORK_MAC, gateway.api.config.mac)},
|
|
|
|
identifiers=set(),
|
|
|
|
)
|
2021-11-14 18:47:15 +00:00
|
|
|
if gateway_host and gateway_host.id in devices_to_be_removed:
|
2020-10-17 16:44:23 +00:00
|
|
|
devices_to_be_removed.remove(gateway_host.id)
|
|
|
|
|
|
|
|
# Don't remove the Gateway service entry
|
|
|
|
gateway_service = device_registry.async_get_device(
|
2021-09-18 07:05:08 +00:00
|
|
|
identifiers={(DOMAIN, gateway.api.config.bridge_id)}, connections=set()
|
2020-10-17 16:44:23 +00:00
|
|
|
)
|
2021-11-14 18:47:15 +00:00
|
|
|
if gateway_service and gateway_service.id in devices_to_be_removed:
|
2020-10-17 16:44:23 +00:00
|
|
|
devices_to_be_removed.remove(gateway_service.id)
|
2020-09-30 18:11:41 +00:00
|
|
|
|
|
|
|
# Don't remove devices belonging to available events
|
|
|
|
for event in gateway.events:
|
|
|
|
if event.device_id in devices_to_be_removed:
|
|
|
|
devices_to_be_removed.remove(event.device_id)
|
|
|
|
|
|
|
|
for entry in entity_entries:
|
|
|
|
|
|
|
|
# Don't remove available entities
|
|
|
|
if entry.unique_id in gateway.entities[entry.domain]:
|
|
|
|
|
|
|
|
# Don't remove devices with available entities
|
|
|
|
if entry.device_id in devices_to_be_removed:
|
|
|
|
devices_to_be_removed.remove(entry.device_id)
|
|
|
|
continue
|
|
|
|
# Remove entities that are not available
|
|
|
|
entities_to_be_removed.append(entry.entity_id)
|
|
|
|
|
|
|
|
# Remove unavailable entities
|
|
|
|
for entity_id in entities_to_be_removed:
|
|
|
|
entity_registry.async_remove(entity_id)
|
|
|
|
|
|
|
|
# Remove devices that don't belong to any entity
|
|
|
|
for device_id in devices_to_be_removed:
|
2020-11-27 08:03:44 +00:00
|
|
|
if (
|
|
|
|
len(
|
|
|
|
async_entries_for_device(
|
|
|
|
entity_registry, device_id, include_disabled_entities=True
|
|
|
|
)
|
|
|
|
)
|
|
|
|
== 0
|
|
|
|
):
|
2020-09-30 18:11:41 +00:00
|
|
|
device_registry.async_remove_device(device_id)
|