core/homeassistant/components/sensor/zwave.py

188 lines
5.6 KiB
Python
Raw Normal View History

2015-02-26 07:27:17 +00:00
"""
homeassistant.components.sensor.zwave
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
Interfaces with Z-Wave sensors.
2015-10-20 19:27:40 +00:00
For more details about the zwave component, please refer to the documentation
at https://home-assistant.io/components/zwave.html
2015-02-26 07:27:17 +00:00
"""
2015-03-01 07:02:26 +00:00
# pylint: disable=import-error
from homeassistant.helpers.event import track_point_in_time
2015-03-01 06:49:55 +00:00
from openzwave.network import ZWaveNetwork
from pydispatch import dispatcher
import datetime
import homeassistant.util.dt as dt_util
import homeassistant.components.zwave as zwave
from homeassistant.helpers.entity import Entity
from homeassistant.const import (
ATTR_BATTERY_LEVEL, STATE_ON, STATE_OFF,
2015-03-01 06:49:55 +00:00
TEMP_CELCIUS, TEMP_FAHRENHEIT, ATTR_LOCATION)
PHILIO = '013c'
PHILIO_SLIM_SENSOR = '0002'
PHILIO_SLIM_SENSOR_MOTION = (PHILIO, PHILIO_SLIM_SENSOR, 0)
2015-10-30 14:28:06 +00:00
WORKAROUND_NO_OFF_EVENT = 'trigger_no_off_event'
2015-10-30 14:28:06 +00:00
DEVICE_MAPPINGS = {
PHILIO_SLIM_SENSOR_MOTION: WORKAROUND_NO_OFF_EVENT,
}
2015-03-01 09:35:58 +00:00
def setup_platform(hass, config, add_devices, discovery_info=None):
""" Sets up Z-Wave sensors. """
node = zwave.NETWORK.nodes[discovery_info[zwave.ATTR_NODE_ID]]
value = node.values[discovery_info[zwave.ATTR_VALUE_ID]]
value.set_change_verified(False)
2015-10-30 14:28:06 +00:00
# if 1 in groups and (zwave.NETWORK.controller.node_id not in
# groups[1].associations):
# node.groups[1].add_association(zwave.NETWORK.controller.node_id)
specific_sensor_key = (value.node.manufacturer_id,
value.node.product_id,
value.index)
# Check workaround mappings for specific devices
2015-10-30 14:28:06 +00:00
if specific_sensor_key in DEVICE_MAPPINGS:
if DEVICE_MAPPINGS[specific_sensor_key] == WORKAROUND_NO_OFF_EVENT:
add_devices([ZWaveTriggerSensor(value, hass)])
return
# generic Device mappings
2015-03-01 09:35:58 +00:00
if value.command_class == zwave.COMMAND_CLASS_SENSOR_BINARY:
2015-03-15 07:01:18 +00:00
add_devices([ZWaveBinarySensor(value)])
2015-03-01 09:35:58 +00:00
elif value.command_class == zwave.COMMAND_CLASS_SENSOR_MULTILEVEL:
2015-03-15 07:01:18 +00:00
add_devices([ZWaveMultilevelSensor(value)])
2015-03-01 09:35:58 +00:00
class ZWaveSensor(Entity):
2015-02-26 07:27:17 +00:00
""" Represents a Z-Wave sensor. """
2015-02-26 07:27:17 +00:00
def __init__(self, sensor_value):
self._value = sensor_value
self._node = sensor_value.node
2015-03-01 06:49:55 +00:00
dispatcher.connect(
self.value_changed, ZWaveNetwork.SIGNAL_VALUE_CHANGED)
2015-03-01 06:49:55 +00:00
@property
def should_poll(self):
""" False because we will push our own state to HA when changed. """
return False
@property
def unique_id(self):
""" Returns a unique id. """
2015-02-26 07:27:17 +00:00
return "ZWAVE-{}-{}".format(self._node.node_id, self._value.object_id)
@property
def name(self):
""" Returns the name of the device. """
name = self._node.name or "{} {}".format(
self._node.manufacturer_name, self._node.product_name)
return "{} {}".format(name, self._value.label)
@property
def state(self):
""" Returns the state of the sensor. """
return self._value.data
@property
def state_attributes(self):
""" Returns the state attributes. """
attrs = {
2015-02-26 07:27:17 +00:00
zwave.ATTR_NODE_ID: self._node.node_id,
}
2015-02-26 07:27:17 +00:00
battery_level = self._node.get_battery_level()
if battery_level is not None:
attrs[ATTR_BATTERY_LEVEL] = battery_level
location = self._node.location
if location:
attrs[ATTR_LOCATION] = location
return attrs
@property
def unit_of_measurement(self):
2015-02-26 07:27:17 +00:00
return self._value.units
def value_changed(self, value):
2015-03-01 06:49:55 +00:00
""" Called when a value has changed on the network. """
if self._value.value_id == value.value_id:
self.update_ha_state()
2015-02-26 07:27:17 +00:00
# pylint: disable=too-few-public-methods
class ZWaveBinarySensor(ZWaveSensor):
""" Represents a binary sensor within Z-Wave. """
2015-02-26 07:27:17 +00:00
@property
def state(self):
""" Returns the state of the sensor. """
return STATE_ON if self._value.data else STATE_OFF
class ZWaveTriggerSensor(ZWaveSensor):
""" Represents a stateless sensor which triggers events within Z-Wave. """
def __init__(self, sensor_value, hass):
super(ZWaveTriggerSensor, self).__init__(sensor_value)
self._hass = hass
self.invalidate_after = None
def value_changed(self, value):
""" Called when a value has changed on the network. """
if self._value.value_id == value.value_id:
self.update_ha_state()
if value.data:
# only allow this value to be true for 60 secs
self.invalidate_after = dt_util.utcnow() + datetime.timedelta(
seconds=60)
track_point_in_time(
self._hass, self.update_ha_state,
self.invalidate_after)
@property
def state(self):
""" Returns the state of the sensor. """
if not self._value.data or \
(self.invalidate_after is not None and
self.invalidate_after <= dt_util.utcnow()):
return STATE_OFF
return STATE_ON
2015-02-26 07:27:17 +00:00
class ZWaveMultilevelSensor(ZWaveSensor):
""" Represents a multi level sensor Z-Wave sensor. """
@property
def state(self):
""" Returns the state of the sensor. """
2015-02-26 07:27:17 +00:00
value = self._value.data
if self._value.units in ('C', 'F'):
return round(value, 1)
2015-03-19 02:15:48 +00:00
elif isinstance(value, float):
return round(value, 2)
2015-02-26 07:27:17 +00:00
return value
@property
def unit_of_measurement(self):
unit = self._value.units
if unit == 'C':
return TEMP_CELCIUS
elif unit == 'F':
return TEMP_FAHRENHEIT
else:
2015-02-26 07:27:17 +00:00
return unit