2016-02-20 20:24:45 +00:00
|
|
|
"""
|
|
|
|
Interfaces with Z-Wave sensors.
|
|
|
|
|
|
|
|
For more details about this platform, please refer to the documentation
|
2016-02-22 09:11:46 +00:00
|
|
|
https://home-assistant.io/components/binary_sensor.zwave/
|
2016-02-20 20:24:45 +00:00
|
|
|
"""
|
|
|
|
import logging
|
2016-02-24 09:28:49 +00:00
|
|
|
import datetime
|
|
|
|
import homeassistant.util.dt as dt_util
|
|
|
|
from homeassistant.helpers.event import track_point_in_time
|
2016-04-17 21:46:51 +00:00
|
|
|
from homeassistant.components import zwave
|
2017-02-08 04:37:11 +00:00
|
|
|
from homeassistant.components.zwave import workaround
|
2017-02-23 21:06:28 +00:00
|
|
|
from homeassistant.components.zwave import async_setup_platform # noqa # pylint: disable=unused-import
|
2016-02-20 20:24:45 +00:00
|
|
|
from homeassistant.components.binary_sensor import (
|
|
|
|
DOMAIN,
|
|
|
|
BinarySensorDevice)
|
|
|
|
|
|
|
|
_LOGGER = logging.getLogger(__name__)
|
|
|
|
DEPENDENCIES = []
|
|
|
|
|
|
|
|
|
2017-03-14 23:55:33 +00:00
|
|
|
def get_device(values, **kwargs):
|
2017-04-30 05:04:49 +00:00
|
|
|
"""Create Z-Wave entity device."""
|
2017-03-14 23:55:33 +00:00
|
|
|
device_mapping = workaround.get_device_mapping(values.primary)
|
2017-02-08 04:37:11 +00:00
|
|
|
if device_mapping == workaround.WORKAROUND_NO_OFF_EVENT:
|
2017-08-06 15:31:32 +00:00
|
|
|
return ZWaveTriggerSensor(values, "motion")
|
2016-02-29 10:05:11 +00:00
|
|
|
|
2017-03-14 23:55:33 +00:00
|
|
|
if workaround.get_device_component_mapping(values.primary) == DOMAIN:
|
|
|
|
return ZWaveBinarySensor(values, None)
|
2016-02-29 10:05:11 +00:00
|
|
|
|
2017-03-14 23:55:33 +00:00
|
|
|
if values.primary.command_class == zwave.const.COMMAND_CLASS_SENSOR_BINARY:
|
|
|
|
return ZWaveBinarySensor(values, None)
|
2017-02-23 21:06:28 +00:00
|
|
|
return None
|
2016-02-20 20:24:45 +00:00
|
|
|
|
|
|
|
|
2017-01-27 06:21:33 +00:00
|
|
|
class ZWaveBinarySensor(BinarySensorDevice, zwave.ZWaveDeviceEntity):
|
2016-03-09 10:15:04 +00:00
|
|
|
"""Representation of a binary sensor within Z-Wave."""
|
2016-02-20 20:24:45 +00:00
|
|
|
|
2017-03-14 23:55:33 +00:00
|
|
|
def __init__(self, values, device_class):
|
2016-03-09 10:15:04 +00:00
|
|
|
"""Initialize the sensor."""
|
2017-03-14 23:55:33 +00:00
|
|
|
zwave.ZWaveDeviceEntity.__init__(self, values, DOMAIN)
|
2017-02-18 07:56:05 +00:00
|
|
|
self._sensor_type = device_class
|
2017-03-14 23:55:33 +00:00
|
|
|
self._state = self.values.primary.data
|
2017-02-18 07:56:05 +00:00
|
|
|
|
|
|
|
def update_properties(self):
|
2017-04-30 05:04:49 +00:00
|
|
|
"""Handle data changes for node values."""
|
2017-03-14 23:55:33 +00:00
|
|
|
self._state = self.values.primary.data
|
2016-02-20 20:24:45 +00:00
|
|
|
|
|
|
|
@property
|
|
|
|
def is_on(self):
|
2017-04-30 05:04:49 +00:00
|
|
|
"""Return true if the binary sensor is on."""
|
2017-02-18 07:56:05 +00:00
|
|
|
return self._state
|
2016-02-20 20:24:45 +00:00
|
|
|
|
|
|
|
@property
|
2017-02-11 04:46:15 +00:00
|
|
|
def device_class(self):
|
|
|
|
"""Return the class of this sensor, from DEVICE_CLASSES."""
|
2016-02-20 20:24:45 +00:00
|
|
|
return self._sensor_type
|
|
|
|
|
2016-02-24 09:28:49 +00:00
|
|
|
|
2017-01-27 06:21:33 +00:00
|
|
|
class ZWaveTriggerSensor(ZWaveBinarySensor):
|
2016-03-09 10:15:04 +00:00
|
|
|
"""Representation of a stateless sensor within Z-Wave."""
|
2016-02-24 09:28:49 +00:00
|
|
|
|
2017-08-06 15:31:32 +00:00
|
|
|
def __init__(self, values, device_class):
|
2016-03-09 10:15:04 +00:00
|
|
|
"""Initialize the sensor."""
|
2017-03-14 23:55:33 +00:00
|
|
|
super(ZWaveTriggerSensor, self).__init__(values, device_class)
|
2017-08-06 15:31:32 +00:00
|
|
|
# Set default off delay to 60 sec
|
|
|
|
self.re_arm_sec = 60
|
2017-02-23 21:06:28 +00:00
|
|
|
self.invalidate_after = None
|
2016-02-24 09:28:49 +00:00
|
|
|
|
2017-02-18 07:56:05 +00:00
|
|
|
def update_properties(self):
|
2017-04-30 05:04:49 +00:00
|
|
|
"""Handle value changes for this entity's node."""
|
2017-03-14 23:55:33 +00:00
|
|
|
self._state = self.values.primary.data
|
2017-08-06 15:31:32 +00:00
|
|
|
_LOGGER.debug('off_delay=%s', self.values.off_delay)
|
|
|
|
# Set re_arm_sec if off_delay is provided from the sensor
|
|
|
|
if self.values.off_delay:
|
|
|
|
_LOGGER.debug('off_delay.data=%s', self.values.off_delay.data)
|
|
|
|
self.re_arm_sec = self.values.off_delay.data * 8
|
2017-02-18 07:56:05 +00:00
|
|
|
# only allow this value to be true for re_arm secs
|
2017-02-23 21:06:28 +00:00
|
|
|
if not self.hass:
|
|
|
|
return
|
|
|
|
|
2017-02-18 07:56:05 +00:00
|
|
|
self.invalidate_after = dt_util.utcnow() + datetime.timedelta(
|
|
|
|
seconds=self.re_arm_sec)
|
|
|
|
track_point_in_time(
|
2017-02-23 21:06:28 +00:00
|
|
|
self.hass, self.async_update_ha_state,
|
2017-02-18 07:56:05 +00:00
|
|
|
self.invalidate_after)
|
2016-02-24 09:28:49 +00:00
|
|
|
|
|
|
|
@property
|
|
|
|
def is_on(self):
|
2017-04-30 05:04:49 +00:00
|
|
|
"""Return true if movement has happened within the rearm time."""
|
2017-02-18 07:56:05 +00:00
|
|
|
return self._state and \
|
2016-02-24 21:44:49 +00:00
|
|
|
(self.invalidate_after is None or
|
|
|
|
self.invalidate_after > dt_util.utcnow())
|