2019-04-03 15:40:03 +00:00
|
|
|
"""Support for exposing NX584 elements as sensors."""
|
2016-02-15 23:07:31 +00:00
|
|
|
import logging
|
|
|
|
import threading
|
|
|
|
import time
|
|
|
|
|
|
|
|
import requests
|
2016-09-30 02:07:35 +00:00
|
|
|
import voluptuous as vol
|
2016-02-15 23:07:31 +00:00
|
|
|
|
2016-02-19 05:27:50 +00:00
|
|
|
from homeassistant.components.binary_sensor import (
|
2019-07-31 19:25:30 +00:00
|
|
|
DEVICE_CLASSES,
|
|
|
|
BinarySensorDevice,
|
|
|
|
PLATFORM_SCHEMA,
|
|
|
|
)
|
|
|
|
from homeassistant.const import CONF_HOST, CONF_PORT
|
2016-09-30 02:07:35 +00:00
|
|
|
import homeassistant.helpers.config_validation as cv
|
2016-02-15 23:07:31 +00:00
|
|
|
|
|
|
|
_LOGGER = logging.getLogger(__name__)
|
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
CONF_EXCLUDE_ZONES = "exclude_zones"
|
|
|
|
CONF_ZONE_TYPES = "zone_types"
|
2016-02-15 23:07:31 +00:00
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
DEFAULT_HOST = "localhost"
|
|
|
|
DEFAULT_PORT = "5007"
|
2016-09-30 02:07:35 +00:00
|
|
|
DEFAULT_SSL = False
|
2016-02-15 23:07:31 +00:00
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
ZONE_TYPES_SCHEMA = vol.Schema({cv.positive_int: vol.In(DEVICE_CLASSES)})
|
2016-02-15 23:07:31 +00:00
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
PLATFORM_SCHEMA = PLATFORM_SCHEMA.extend(
|
|
|
|
{
|
|
|
|
vol.Optional(CONF_EXCLUDE_ZONES, default=[]): vol.All(
|
|
|
|
cv.ensure_list, [cv.positive_int]
|
|
|
|
),
|
|
|
|
vol.Optional(CONF_HOST, default=DEFAULT_HOST): cv.string,
|
|
|
|
vol.Optional(CONF_PORT, default=DEFAULT_PORT): cv.port,
|
|
|
|
vol.Optional(CONF_ZONE_TYPES, default={}): ZONE_TYPES_SCHEMA,
|
|
|
|
}
|
|
|
|
)
|
2016-02-15 23:07:31 +00:00
|
|
|
|
2016-09-30 02:07:35 +00:00
|
|
|
|
2018-08-24 14:37:30 +00:00
|
|
|
def setup_platform(hass, config, add_entities, discovery_info=None):
|
2017-04-30 05:04:49 +00:00
|
|
|
"""Set up the NX584 binary sensor platform."""
|
2016-09-30 02:07:35 +00:00
|
|
|
from nx584 import client as nx584_client
|
|
|
|
|
|
|
|
host = config.get(CONF_HOST)
|
|
|
|
port = config.get(CONF_PORT)
|
|
|
|
exclude = config.get(CONF_EXCLUDE_ZONES)
|
|
|
|
zone_types = config.get(CONF_ZONE_TYPES)
|
2016-02-18 00:00:01 +00:00
|
|
|
|
2016-02-15 23:07:31 +00:00
|
|
|
try:
|
2019-09-03 18:35:00 +00:00
|
|
|
client = nx584_client.Client(f"http://{host}:{port}")
|
2016-02-15 23:07:31 +00:00
|
|
|
zones = client.list_zones()
|
|
|
|
except requests.exceptions.ConnectionError as ex:
|
2017-04-30 05:04:49 +00:00
|
|
|
_LOGGER.error("Unable to connect to NX584: %s", str(ex))
|
2016-02-15 23:07:31 +00:00
|
|
|
return False
|
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
version = [int(v) for v in client.get_version().split(".")]
|
2016-02-15 23:07:31 +00:00
|
|
|
if version < [1, 1]:
|
2016-09-30 02:07:35 +00:00
|
|
|
_LOGGER.error("NX584 is too old to use for sensors (>=0.2 required)")
|
2016-02-15 23:07:31 +00:00
|
|
|
return False
|
|
|
|
|
|
|
|
zone_sensors = {
|
2019-07-31 19:25:30 +00:00
|
|
|
zone["number"]: NX584ZoneSensor(zone, zone_types.get(zone["number"], "opening"))
|
2016-02-15 23:07:31 +00:00
|
|
|
for zone in zones
|
2019-07-31 19:25:30 +00:00
|
|
|
if zone["number"] not in exclude
|
|
|
|
}
|
2016-02-15 23:07:31 +00:00
|
|
|
if zone_sensors:
|
2018-08-24 14:37:30 +00:00
|
|
|
add_entities(zone_sensors.values())
|
2016-02-15 23:07:31 +00:00
|
|
|
watcher = NX584Watcher(client, zone_sensors)
|
|
|
|
watcher.start()
|
|
|
|
else:
|
2016-09-30 02:07:35 +00:00
|
|
|
_LOGGER.warning("No zones found on NX584")
|
2016-02-15 23:07:31 +00:00
|
|
|
return True
|
|
|
|
|
|
|
|
|
|
|
|
class NX584ZoneSensor(BinarySensorDevice):
|
2016-09-30 02:07:35 +00:00
|
|
|
"""Representation of a NX584 zone as a sensor."""
|
2016-02-15 23:07:31 +00:00
|
|
|
|
2016-02-18 00:00:01 +00:00
|
|
|
def __init__(self, zone, zone_type):
|
2016-03-07 19:21:08 +00:00
|
|
|
"""Initialize the nx594 binary sensor."""
|
2016-02-15 23:07:31 +00:00
|
|
|
self._zone = zone
|
2016-02-18 00:00:01 +00:00
|
|
|
self._zone_type = zone_type
|
|
|
|
|
|
|
|
@property
|
2017-02-11 04:46:15 +00:00
|
|
|
def device_class(self):
|
|
|
|
"""Return the class of this sensor, from DEVICE_CLASSES."""
|
2016-02-18 00:00:01 +00:00
|
|
|
return self._zone_type
|
2016-02-15 23:07:31 +00:00
|
|
|
|
|
|
|
@property
|
|
|
|
def should_poll(self):
|
2016-02-22 09:11:46 +00:00
|
|
|
"""No polling needed."""
|
2016-02-15 23:07:31 +00:00
|
|
|
return False
|
|
|
|
|
|
|
|
@property
|
|
|
|
def name(self):
|
2016-03-07 19:21:08 +00:00
|
|
|
"""Return the name of the binary sensor."""
|
2019-07-31 19:25:30 +00:00
|
|
|
return self._zone["name"]
|
2016-02-15 23:07:31 +00:00
|
|
|
|
|
|
|
@property
|
|
|
|
def is_on(self):
|
2016-02-22 09:11:46 +00:00
|
|
|
"""Return true if the binary sensor is on."""
|
2016-02-15 23:07:31 +00:00
|
|
|
# True means "faulted" or "open" or "abnormal state"
|
2019-07-31 19:25:30 +00:00
|
|
|
return self._zone["state"]
|
2016-02-15 23:07:31 +00:00
|
|
|
|
|
|
|
|
|
|
|
class NX584Watcher(threading.Thread):
|
|
|
|
"""Event listener thread to process NX584 events."""
|
|
|
|
|
|
|
|
def __init__(self, client, zone_sensors):
|
2016-09-30 02:07:35 +00:00
|
|
|
"""Initialize NX584 watcher thread."""
|
2019-09-24 22:38:20 +00:00
|
|
|
super().__init__()
|
2016-02-15 23:07:31 +00:00
|
|
|
self.daemon = True
|
|
|
|
self._client = client
|
|
|
|
self._zone_sensors = zone_sensors
|
|
|
|
|
|
|
|
def _process_zone_event(self, event):
|
2019-07-31 19:25:30 +00:00
|
|
|
zone = event["zone"]
|
2016-02-15 23:07:31 +00:00
|
|
|
zone_sensor = self._zone_sensors.get(zone)
|
|
|
|
# pylint: disable=protected-access
|
|
|
|
if not zone_sensor:
|
|
|
|
return
|
2019-07-31 19:25:30 +00:00
|
|
|
zone_sensor._zone["state"] = event["zone_state"]
|
2016-11-17 15:34:46 +00:00
|
|
|
zone_sensor.schedule_update_ha_state()
|
2016-02-15 23:07:31 +00:00
|
|
|
|
|
|
|
def _process_events(self, events):
|
|
|
|
for event in events:
|
2019-07-31 19:25:30 +00:00
|
|
|
if event.get("type") == "zone_status":
|
2016-02-15 23:07:31 +00:00
|
|
|
self._process_zone_event(event)
|
|
|
|
|
|
|
|
def _run(self):
|
2016-03-07 19:21:08 +00:00
|
|
|
"""Throw away any existing events so we don't replay history."""
|
2016-02-15 23:07:31 +00:00
|
|
|
self._client.get_events()
|
|
|
|
while True:
|
|
|
|
events = self._client.get_events()
|
|
|
|
if events:
|
|
|
|
self._process_events(events)
|
|
|
|
|
|
|
|
def run(self):
|
2016-03-07 19:21:08 +00:00
|
|
|
"""Run the watcher."""
|
2016-02-15 23:07:31 +00:00
|
|
|
while True:
|
|
|
|
try:
|
|
|
|
self._run()
|
|
|
|
except requests.exceptions.ConnectionError:
|
2016-09-30 02:07:35 +00:00
|
|
|
_LOGGER.error("Failed to reach NX584 server")
|
2016-02-15 23:07:31 +00:00
|
|
|
time.sleep(10)
|