2020-05-14 20:56:04 +00:00
|
|
|
"""The ozw integration."""
|
2020-05-03 00:54:16 +00:00
|
|
|
import asyncio
|
|
|
|
import json
|
|
|
|
import logging
|
|
|
|
|
|
|
|
from openzwavemqtt import OZWManager, OZWOptions
|
|
|
|
from openzwavemqtt.const import (
|
|
|
|
EVENT_INSTANCE_EVENT,
|
|
|
|
EVENT_NODE_ADDED,
|
|
|
|
EVENT_NODE_CHANGED,
|
|
|
|
EVENT_NODE_REMOVED,
|
|
|
|
EVENT_VALUE_ADDED,
|
|
|
|
EVENT_VALUE_CHANGED,
|
|
|
|
EVENT_VALUE_REMOVED,
|
|
|
|
CommandClass,
|
|
|
|
ValueType,
|
|
|
|
)
|
|
|
|
from openzwavemqtt.models.node import OZWNode
|
|
|
|
from openzwavemqtt.models.value import OZWValue
|
|
|
|
import voluptuous as vol
|
|
|
|
|
|
|
|
from homeassistant.components import mqtt
|
|
|
|
from homeassistant.config_entries import ConfigEntry
|
|
|
|
from homeassistant.core import HomeAssistant, callback
|
|
|
|
from homeassistant.helpers.device_registry import async_get_registry as get_dev_reg
|
|
|
|
from homeassistant.helpers.dispatcher import async_dispatcher_send
|
|
|
|
|
|
|
|
from . import const
|
2020-08-12 19:49:40 +00:00
|
|
|
from .const import (
|
|
|
|
DATA_UNSUBSCRIBE,
|
|
|
|
DOMAIN,
|
|
|
|
MANAGER,
|
|
|
|
OPTIONS,
|
|
|
|
PLATFORMS,
|
|
|
|
TOPIC_OPENZWAVE,
|
|
|
|
)
|
2020-05-03 00:54:16 +00:00
|
|
|
from .discovery import DISCOVERY_SCHEMAS, check_node_schema, check_value_schema
|
|
|
|
from .entity import (
|
|
|
|
ZWaveDeviceEntityValues,
|
|
|
|
create_device_id,
|
|
|
|
create_device_name,
|
|
|
|
create_value_id,
|
|
|
|
)
|
|
|
|
from .services import ZWaveServices
|
2020-08-12 19:49:40 +00:00
|
|
|
from .websocket_api import async_register_api
|
2020-05-03 00:54:16 +00:00
|
|
|
|
|
|
|
_LOGGER = logging.getLogger(__name__)
|
|
|
|
|
|
|
|
CONFIG_SCHEMA = vol.Schema({DOMAIN: vol.Schema({})}, extra=vol.ALLOW_EXTRA)
|
|
|
|
DATA_DEVICES = "zwave-mqtt-devices"
|
|
|
|
|
|
|
|
|
|
|
|
async def async_setup(hass: HomeAssistant, config: dict):
|
2020-05-14 20:56:04 +00:00
|
|
|
"""Initialize basic config of ozw component."""
|
2020-05-03 00:54:16 +00:00
|
|
|
if "mqtt" not in hass.config.components:
|
|
|
|
_LOGGER.error("MQTT integration is not set up")
|
|
|
|
return False
|
|
|
|
hass.data[DOMAIN] = {}
|
|
|
|
return True
|
|
|
|
|
|
|
|
|
|
|
|
async def async_setup_entry(hass: HomeAssistant, entry: ConfigEntry):
|
2020-05-14 20:56:04 +00:00
|
|
|
"""Set up ozw from a config entry."""
|
|
|
|
ozw_data = hass.data[DOMAIN][entry.entry_id] = {}
|
|
|
|
ozw_data[DATA_UNSUBSCRIBE] = []
|
2020-05-03 00:54:16 +00:00
|
|
|
|
|
|
|
data_nodes = {}
|
|
|
|
data_values = {}
|
|
|
|
removed_nodes = []
|
|
|
|
|
|
|
|
@callback
|
|
|
|
def send_message(topic, payload):
|
|
|
|
mqtt.async_publish(hass, topic, json.dumps(payload))
|
|
|
|
|
|
|
|
options = OZWOptions(send_message=send_message, topic_prefix=f"{TOPIC_OPENZWAVE}/")
|
|
|
|
manager = OZWManager(options)
|
|
|
|
|
2020-08-12 19:49:40 +00:00
|
|
|
hass.data[DOMAIN][MANAGER] = manager
|
|
|
|
hass.data[DOMAIN][OPTIONS] = options
|
|
|
|
|
2020-05-03 00:54:16 +00:00
|
|
|
@callback
|
|
|
|
def async_node_added(node):
|
|
|
|
# Caution: This is also called on (re)start.
|
|
|
|
_LOGGER.debug("[NODE ADDED] node_id: %s", node.id)
|
|
|
|
data_nodes[node.id] = node
|
|
|
|
if node.id not in data_values:
|
|
|
|
data_values[node.id] = []
|
|
|
|
|
|
|
|
@callback
|
|
|
|
def async_node_changed(node):
|
|
|
|
_LOGGER.debug("[NODE CHANGED] node_id: %s", node.id)
|
|
|
|
data_nodes[node.id] = node
|
|
|
|
# notify devices about the node change
|
|
|
|
if node.id not in removed_nodes:
|
|
|
|
hass.async_create_task(async_handle_node_update(hass, node))
|
|
|
|
|
|
|
|
@callback
|
|
|
|
def async_node_removed(node):
|
|
|
|
_LOGGER.debug("[NODE REMOVED] node_id: %s", node.id)
|
|
|
|
data_nodes.pop(node.id)
|
|
|
|
# node added/removed events also happen on (re)starts of hass/mqtt/ozw
|
|
|
|
# cleanup device/entity registry if we know this node is permanently deleted
|
|
|
|
# entities itself are removed by the values logic
|
|
|
|
if node.id in removed_nodes:
|
|
|
|
hass.async_create_task(async_handle_remove_node(hass, node))
|
|
|
|
removed_nodes.remove(node.id)
|
|
|
|
|
|
|
|
@callback
|
|
|
|
def async_instance_event(message):
|
|
|
|
event = message["event"]
|
|
|
|
event_data = message["data"]
|
|
|
|
_LOGGER.debug("[INSTANCE EVENT]: %s - data: %s", event, event_data)
|
|
|
|
# The actual removal action of a Z-Wave node is reported as instance event
|
|
|
|
# Only when this event is detected we cleanup the device and entities from hass
|
2020-07-08 07:52:03 +00:00
|
|
|
# Note: Find a more elegant way of doing this, e.g. a notification of this event from OZW
|
|
|
|
if event in ["removenode", "removefailednode"] and "Node" in event_data:
|
2020-05-03 00:54:16 +00:00
|
|
|
removed_nodes.append(event_data["Node"])
|
|
|
|
|
|
|
|
@callback
|
|
|
|
def async_value_added(value):
|
|
|
|
node = value.node
|
2020-05-07 21:52:54 +00:00
|
|
|
# Clean up node.node_id and node.id use. They are the same.
|
2020-05-03 00:54:16 +00:00
|
|
|
node_id = value.node.node_id
|
|
|
|
|
|
|
|
# Filter out CommandClasses we're definitely not interested in.
|
|
|
|
if value.command_class in [
|
|
|
|
CommandClass.MANUFACTURER_SPECIFIC,
|
|
|
|
]:
|
|
|
|
return
|
|
|
|
|
|
|
|
_LOGGER.debug(
|
|
|
|
"[VALUE ADDED] node_id: %s - label: %s - value: %s - value_id: %s - CC: %s",
|
|
|
|
value.node.id,
|
|
|
|
value.label,
|
|
|
|
value.value,
|
|
|
|
value.value_id_key,
|
|
|
|
value.command_class,
|
|
|
|
)
|
|
|
|
|
|
|
|
node_data_values = data_values[node_id]
|
|
|
|
|
|
|
|
# Check if this value should be tracked by an existing entity
|
|
|
|
value_unique_id = create_value_id(value)
|
|
|
|
for values in node_data_values:
|
|
|
|
values.async_check_value(value)
|
|
|
|
if values.values_id == value_unique_id:
|
|
|
|
return # this value already has an entity
|
|
|
|
|
|
|
|
# Run discovery on it and see if any entities need created
|
|
|
|
for schema in DISCOVERY_SCHEMAS:
|
|
|
|
if not check_node_schema(node, schema):
|
|
|
|
continue
|
|
|
|
if not check_value_schema(
|
|
|
|
value, schema[const.DISC_VALUES][const.DISC_PRIMARY]
|
|
|
|
):
|
|
|
|
continue
|
|
|
|
|
|
|
|
values = ZWaveDeviceEntityValues(hass, options, schema, value)
|
|
|
|
values.async_setup()
|
|
|
|
|
2020-05-07 21:52:54 +00:00
|
|
|
# This is legacy and can be cleaned up since we are in the main thread:
|
2020-05-03 00:54:16 +00:00
|
|
|
# We create a new list and update the reference here so that
|
|
|
|
# the list can be safely iterated over in the main thread
|
|
|
|
data_values[node_id] = node_data_values + [values]
|
|
|
|
|
|
|
|
@callback
|
|
|
|
def async_value_changed(value):
|
|
|
|
# if an entity belonging to this value needs updating,
|
|
|
|
# it's handled within the entity logic
|
|
|
|
_LOGGER.debug(
|
|
|
|
"[VALUE CHANGED] node_id: %s - label: %s - value: %s - value_id: %s - CC: %s",
|
|
|
|
value.node.id,
|
|
|
|
value.label,
|
|
|
|
value.value,
|
|
|
|
value.value_id_key,
|
|
|
|
value.command_class,
|
|
|
|
)
|
|
|
|
# Handle a scene activation message
|
|
|
|
if value.command_class in [
|
|
|
|
CommandClass.SCENE_ACTIVATION,
|
|
|
|
CommandClass.CENTRAL_SCENE,
|
|
|
|
]:
|
|
|
|
async_handle_scene_activated(hass, value)
|
|
|
|
return
|
|
|
|
|
|
|
|
@callback
|
|
|
|
def async_value_removed(value):
|
|
|
|
_LOGGER.debug(
|
|
|
|
"[VALUE REMOVED] node_id: %s - label: %s - value: %s - value_id: %s - CC: %s",
|
|
|
|
value.node.id,
|
|
|
|
value.label,
|
|
|
|
value.value,
|
|
|
|
value.value_id_key,
|
|
|
|
value.command_class,
|
|
|
|
)
|
|
|
|
# signal all entities using this value for removal
|
|
|
|
value_unique_id = create_value_id(value)
|
|
|
|
async_dispatcher_send(hass, const.SIGNAL_DELETE_ENTITY, value_unique_id)
|
|
|
|
# remove value from our local list
|
|
|
|
node_data_values = data_values[value.node.id]
|
|
|
|
node_data_values[:] = [
|
|
|
|
item for item in node_data_values if item.values_id != value_unique_id
|
|
|
|
]
|
|
|
|
|
|
|
|
# Listen to events for node and value changes
|
2020-08-12 13:17:21 +00:00
|
|
|
for event, event_callback in (
|
|
|
|
(EVENT_NODE_ADDED, async_node_added),
|
|
|
|
(EVENT_NODE_CHANGED, async_node_changed),
|
|
|
|
(EVENT_NODE_REMOVED, async_node_removed),
|
|
|
|
(EVENT_VALUE_ADDED, async_value_added),
|
|
|
|
(EVENT_VALUE_CHANGED, async_value_changed),
|
|
|
|
(EVENT_VALUE_REMOVED, async_value_removed),
|
|
|
|
(EVENT_INSTANCE_EVENT, async_instance_event),
|
|
|
|
):
|
|
|
|
ozw_data[DATA_UNSUBSCRIBE].append(options.listen(event, event_callback))
|
2020-05-03 00:54:16 +00:00
|
|
|
|
|
|
|
# Register Services
|
|
|
|
services = ZWaveServices(hass, manager)
|
|
|
|
services.async_register()
|
|
|
|
|
2020-07-29 20:35:26 +00:00
|
|
|
# Register WebSocket API
|
2020-08-12 19:49:40 +00:00
|
|
|
async_register_api(hass)
|
2020-07-29 20:35:26 +00:00
|
|
|
|
2020-05-03 00:54:16 +00:00
|
|
|
@callback
|
|
|
|
def async_receive_message(msg):
|
|
|
|
manager.receive_message(msg.topic, msg.payload)
|
|
|
|
|
|
|
|
async def start_platforms():
|
|
|
|
await asyncio.gather(
|
|
|
|
*[
|
|
|
|
hass.config_entries.async_forward_entry_setup(entry, component)
|
|
|
|
for component in PLATFORMS
|
|
|
|
]
|
|
|
|
)
|
2020-05-14 20:56:04 +00:00
|
|
|
ozw_data[DATA_UNSUBSCRIBE].append(
|
2020-05-03 00:54:16 +00:00
|
|
|
await mqtt.async_subscribe(
|
|
|
|
hass, f"{TOPIC_OPENZWAVE}/#", async_receive_message
|
|
|
|
)
|
|
|
|
)
|
|
|
|
|
|
|
|
hass.async_create_task(start_platforms())
|
|
|
|
|
|
|
|
return True
|
|
|
|
|
|
|
|
|
|
|
|
async def async_unload_entry(hass: HomeAssistant, entry: ConfigEntry):
|
|
|
|
"""Unload a config entry."""
|
|
|
|
# cleanup platforms
|
|
|
|
unload_ok = all(
|
|
|
|
await asyncio.gather(
|
|
|
|
*[
|
|
|
|
hass.config_entries.async_forward_entry_unload(entry, component)
|
|
|
|
for component in PLATFORMS
|
|
|
|
]
|
|
|
|
)
|
|
|
|
)
|
|
|
|
if not unload_ok:
|
|
|
|
return False
|
|
|
|
|
|
|
|
# unsubscribe all listeners
|
|
|
|
for unsubscribe_listener in hass.data[DOMAIN][entry.entry_id][DATA_UNSUBSCRIBE]:
|
|
|
|
unsubscribe_listener()
|
|
|
|
hass.data[DOMAIN].pop(entry.entry_id)
|
|
|
|
|
|
|
|
return True
|
|
|
|
|
|
|
|
|
|
|
|
async def async_handle_remove_node(hass: HomeAssistant, node: OZWNode):
|
|
|
|
"""Handle the removal of a Z-Wave node, removing all traces in device/entity registry."""
|
|
|
|
dev_registry = await get_dev_reg(hass)
|
|
|
|
# grab device in device registry attached to this node
|
|
|
|
dev_id = create_device_id(node)
|
|
|
|
device = dev_registry.async_get_device({(DOMAIN, dev_id)}, set())
|
|
|
|
if not device:
|
|
|
|
return
|
|
|
|
devices_to_remove = [device.id]
|
|
|
|
# also grab slave devices (node instances)
|
|
|
|
for item in dev_registry.devices.values():
|
|
|
|
if item.via_device_id == device.id:
|
|
|
|
devices_to_remove.append(item.id)
|
|
|
|
# remove all devices in registry related to this node
|
|
|
|
# note: removal of entity registry is handled by core
|
|
|
|
for dev_id in devices_to_remove:
|
|
|
|
dev_registry.async_remove_device(dev_id)
|
|
|
|
|
|
|
|
|
|
|
|
async def async_handle_node_update(hass: HomeAssistant, node: OZWNode):
|
|
|
|
"""
|
|
|
|
Handle a node updated event from OZW.
|
|
|
|
|
|
|
|
Meaning some of the basic info like name/model is updated.
|
|
|
|
We want these changes to be pushed to the device registry.
|
|
|
|
"""
|
|
|
|
dev_registry = await get_dev_reg(hass)
|
|
|
|
# grab device in device registry attached to this node
|
|
|
|
dev_id = create_device_id(node)
|
|
|
|
device = dev_registry.async_get_device({(DOMAIN, dev_id)}, set())
|
|
|
|
if not device:
|
|
|
|
return
|
|
|
|
# update device in device registry with (updated) info
|
|
|
|
for item in dev_registry.devices.values():
|
|
|
|
if item.id != device.id and item.via_device_id != device.id:
|
|
|
|
continue
|
|
|
|
dev_name = create_device_name(node)
|
|
|
|
dev_registry.async_update_device(
|
|
|
|
item.id,
|
|
|
|
manufacturer=node.node_manufacturer_name,
|
|
|
|
model=node.node_product_name,
|
|
|
|
name=dev_name,
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
|
|
@callback
|
|
|
|
def async_handle_scene_activated(hass: HomeAssistant, scene_value: OZWValue):
|
|
|
|
"""Handle a (central) scene activation message."""
|
|
|
|
node_id = scene_value.node.id
|
|
|
|
scene_id = scene_value.index
|
|
|
|
scene_label = scene_value.label
|
|
|
|
if scene_value.command_class == CommandClass.SCENE_ACTIVATION:
|
|
|
|
# legacy/network scene
|
|
|
|
scene_value_id = scene_value.value
|
|
|
|
scene_value_label = scene_value.label
|
|
|
|
else:
|
|
|
|
# central scene command
|
|
|
|
if scene_value.type != ValueType.LIST:
|
|
|
|
return
|
|
|
|
scene_value_label = scene_value.value["Selected"]
|
|
|
|
scene_value_id = scene_value.value["Selected_id"]
|
|
|
|
|
|
|
|
_LOGGER.debug(
|
|
|
|
"[SCENE_ACTIVATED] node_id: %s - scene_id: %s - scene_value_id: %s",
|
|
|
|
node_id,
|
|
|
|
scene_id,
|
|
|
|
scene_value_id,
|
|
|
|
)
|
|
|
|
# Simply forward it to the hass event bus
|
|
|
|
hass.bus.async_fire(
|
|
|
|
const.EVENT_SCENE_ACTIVATED,
|
|
|
|
{
|
|
|
|
const.ATTR_NODE_ID: node_id,
|
|
|
|
const.ATTR_SCENE_ID: scene_id,
|
|
|
|
const.ATTR_SCENE_LABEL: scene_label,
|
|
|
|
const.ATTR_SCENE_VALUE_ID: scene_value_id,
|
|
|
|
const.ATTR_SCENE_VALUE_LABEL: scene_value_label,
|
|
|
|
},
|
|
|
|
)
|