2017-06-05 06:53:25 +00:00
|
|
|
"""
|
|
|
|
Support for Vanderbilt (formerly Siemens) SPC alarm systems.
|
|
|
|
|
|
|
|
For more details about this platform, please refer to the documentation at
|
|
|
|
https://home-assistant.io/components/spc/
|
|
|
|
"""
|
|
|
|
import asyncio
|
|
|
|
import json
|
2018-01-21 06:35:38 +00:00
|
|
|
import logging
|
2017-06-05 06:53:25 +00:00
|
|
|
from urllib.parse import urljoin
|
|
|
|
|
|
|
|
import aiohttp
|
|
|
|
import async_timeout
|
|
|
|
import voluptuous as vol
|
|
|
|
|
|
|
|
from homeassistant.const import (
|
2018-01-21 06:35:38 +00:00
|
|
|
STATE_ALARM_ARMED_AWAY, STATE_ALARM_ARMED_HOME, STATE_ALARM_DISARMED,
|
|
|
|
STATE_ALARM_TRIGGERED, STATE_OFF, STATE_ON, STATE_UNAVAILABLE,
|
|
|
|
STATE_UNKNOWN)
|
|
|
|
from homeassistant.helpers import discovery
|
|
|
|
import homeassistant.helpers.config_validation as cv
|
2017-06-05 06:53:25 +00:00
|
|
|
|
|
|
|
REQUIREMENTS = ['websockets==3.2']
|
|
|
|
|
|
|
|
_LOGGER = logging.getLogger(__name__)
|
|
|
|
|
|
|
|
ATTR_DISCOVER_DEVICES = 'devices'
|
|
|
|
ATTR_DISCOVER_AREAS = 'areas'
|
|
|
|
|
|
|
|
CONF_WS_URL = 'ws_url'
|
|
|
|
CONF_API_URL = 'api_url'
|
|
|
|
|
|
|
|
DATA_REGISTRY = 'spc_registry'
|
|
|
|
DATA_API = 'spc_api'
|
2018-01-21 06:35:38 +00:00
|
|
|
DOMAIN = 'spc'
|
2017-06-05 06:53:25 +00:00
|
|
|
|
|
|
|
CONFIG_SCHEMA = vol.Schema({
|
|
|
|
DOMAIN: vol.Schema({
|
|
|
|
vol.Required(CONF_WS_URL): cv.string,
|
|
|
|
vol.Required(CONF_API_URL): cv.string
|
|
|
|
}),
|
|
|
|
}, extra=vol.ALLOW_EXTRA)
|
|
|
|
|
|
|
|
|
|
|
|
@asyncio.coroutine
|
|
|
|
def async_setup(hass, config):
|
2018-01-21 06:35:38 +00:00
|
|
|
"""Set up the SPC platform."""
|
2017-06-05 06:53:25 +00:00
|
|
|
hass.data[DATA_REGISTRY] = SpcRegistry()
|
|
|
|
|
|
|
|
api = SpcWebGateway(hass,
|
|
|
|
config[DOMAIN].get(CONF_API_URL),
|
|
|
|
config[DOMAIN].get(CONF_WS_URL))
|
|
|
|
|
|
|
|
hass.data[DATA_API] = api
|
|
|
|
|
|
|
|
# add sensor devices for each zone (typically motion/fire/door sensors)
|
|
|
|
zones = yield from api.get_zones()
|
|
|
|
if zones:
|
2018-07-23 12:05:38 +00:00
|
|
|
hass.async_create_task(discovery.async_load_platform(
|
2017-06-05 06:53:25 +00:00
|
|
|
hass, 'binary_sensor', DOMAIN,
|
|
|
|
{ATTR_DISCOVER_DEVICES: zones}, config))
|
|
|
|
|
|
|
|
# create a separate alarm panel for each area
|
|
|
|
areas = yield from api.get_areas()
|
|
|
|
if areas:
|
2018-07-23 12:05:38 +00:00
|
|
|
hass.async_create_task(discovery.async_load_platform(
|
2017-06-05 06:53:25 +00:00
|
|
|
hass, 'alarm_control_panel', DOMAIN,
|
|
|
|
{ATTR_DISCOVER_AREAS: areas}, config))
|
|
|
|
|
|
|
|
# start listening for incoming events over websocket
|
|
|
|
api.start_listener(_async_process_message, hass.data[DATA_REGISTRY])
|
|
|
|
|
|
|
|
return True
|
|
|
|
|
|
|
|
|
|
|
|
@asyncio.coroutine
|
|
|
|
def _async_process_message(sia_message, spc_registry):
|
|
|
|
spc_id = sia_message['sia_address']
|
|
|
|
sia_code = sia_message['sia_code']
|
|
|
|
|
|
|
|
# BA - Burglary Alarm
|
|
|
|
# CG - Close Area
|
|
|
|
# NL - Perimeter Armed
|
|
|
|
# OG - Open Area
|
|
|
|
# ZO - Zone Open
|
|
|
|
# ZC - Zone Close
|
|
|
|
# ZX - Zone Short
|
|
|
|
# ZD - Zone Disconnected
|
|
|
|
|
2017-11-11 20:36:03 +00:00
|
|
|
extra = {}
|
|
|
|
|
|
|
|
if sia_code in ('BA', 'CG', 'NL', 'OG'):
|
2017-06-05 06:53:25 +00:00
|
|
|
# change in area status, notify alarm panel device
|
|
|
|
device = spc_registry.get_alarm_device(spc_id)
|
2017-11-11 20:36:03 +00:00
|
|
|
data = sia_message['description'].split('¦')
|
|
|
|
if len(data) == 3:
|
|
|
|
extra['changed_by'] = data[1]
|
2017-06-05 06:53:25 +00:00
|
|
|
else:
|
2018-01-21 06:35:38 +00:00
|
|
|
# Change in zone status, notify sensor device
|
2017-06-05 06:53:25 +00:00
|
|
|
device = spc_registry.get_sensor_device(spc_id)
|
|
|
|
|
2018-01-21 06:35:38 +00:00
|
|
|
sia_code_to_state_map = {
|
|
|
|
'BA': STATE_ALARM_TRIGGERED,
|
|
|
|
'CG': STATE_ALARM_ARMED_AWAY,
|
|
|
|
'NL': STATE_ALARM_ARMED_HOME,
|
|
|
|
'OG': STATE_ALARM_DISARMED,
|
|
|
|
'ZO': STATE_ON,
|
|
|
|
'ZC': STATE_OFF,
|
|
|
|
'ZX': STATE_UNKNOWN,
|
|
|
|
'ZD': STATE_UNAVAILABLE,
|
|
|
|
}
|
2017-06-05 06:53:25 +00:00
|
|
|
|
|
|
|
new_state = sia_code_to_state_map.get(sia_code, None)
|
|
|
|
|
|
|
|
if new_state and not device:
|
2018-01-21 06:35:38 +00:00
|
|
|
_LOGGER.warning(
|
|
|
|
"No device mapping found for SPC area/zone id %s", spc_id)
|
2017-06-05 06:53:25 +00:00
|
|
|
elif new_state:
|
2017-11-11 20:36:03 +00:00
|
|
|
yield from device.async_update_from_spc(new_state, extra)
|
2017-06-05 06:53:25 +00:00
|
|
|
|
|
|
|
|
|
|
|
class SpcRegistry:
|
2018-01-21 06:35:38 +00:00
|
|
|
"""Maintain mappings between SPC zones/areas and HA entities."""
|
2017-06-05 06:53:25 +00:00
|
|
|
|
|
|
|
def __init__(self):
|
|
|
|
"""Initialize the registry."""
|
|
|
|
self._zone_id_to_sensor_map = {}
|
|
|
|
self._area_id_to_alarm_map = {}
|
|
|
|
|
|
|
|
def register_sensor_device(self, zone_id, device):
|
|
|
|
"""Add a sensor device to the registry."""
|
|
|
|
self._zone_id_to_sensor_map[zone_id] = device
|
|
|
|
|
|
|
|
def get_sensor_device(self, zone_id):
|
|
|
|
"""Retrieve a sensor device for a specific zone."""
|
|
|
|
return self._zone_id_to_sensor_map.get(zone_id, None)
|
|
|
|
|
|
|
|
def register_alarm_device(self, area_id, device):
|
|
|
|
"""Add an alarm device to the registry."""
|
|
|
|
self._area_id_to_alarm_map[area_id] = device
|
|
|
|
|
|
|
|
def get_alarm_device(self, area_id):
|
|
|
|
"""Retrieve an alarm device for a specific area."""
|
|
|
|
return self._area_id_to_alarm_map.get(area_id, None)
|
|
|
|
|
|
|
|
|
|
|
|
@asyncio.coroutine
|
|
|
|
def _ws_process_message(message, async_callback, *args):
|
|
|
|
if message.get('status', '') != 'success':
|
2018-01-21 06:35:38 +00:00
|
|
|
_LOGGER.warning(
|
|
|
|
"Unsuccessful websocket message delivered, ignoring: %s", message)
|
2017-06-05 06:53:25 +00:00
|
|
|
try:
|
|
|
|
yield from async_callback(message['data']['sia'], *args)
|
2018-07-17 17:34:29 +00:00
|
|
|
except: # noqa: E722 pylint: disable=bare-except
|
2018-01-21 06:35:38 +00:00
|
|
|
_LOGGER.exception("Exception in callback, ignoring")
|
2017-06-05 06:53:25 +00:00
|
|
|
|
|
|
|
|
|
|
|
class SpcWebGateway:
|
|
|
|
"""Simple binding for the Lundix SPC Web Gateway REST API."""
|
|
|
|
|
|
|
|
AREA_COMMAND_SET = 'set'
|
|
|
|
AREA_COMMAND_PART_SET = 'set_a'
|
|
|
|
AREA_COMMAND_UNSET = 'unset'
|
|
|
|
|
|
|
|
def __init__(self, hass, api_url, ws_url):
|
|
|
|
"""Initialize the web gateway client."""
|
|
|
|
self._hass = hass
|
|
|
|
self._api_url = api_url
|
|
|
|
self._ws_url = ws_url
|
|
|
|
self._ws = None
|
|
|
|
|
|
|
|
@asyncio.coroutine
|
|
|
|
def get_zones(self):
|
|
|
|
"""Retrieve all available zones."""
|
|
|
|
return (yield from self._get_data('zone'))
|
|
|
|
|
|
|
|
@asyncio.coroutine
|
|
|
|
def get_areas(self):
|
|
|
|
"""Retrieve all available areas."""
|
|
|
|
return (yield from self._get_data('area'))
|
|
|
|
|
|
|
|
@asyncio.coroutine
|
|
|
|
def send_area_command(self, area_id, command):
|
|
|
|
"""Send an area command."""
|
2018-01-21 06:35:38 +00:00
|
|
|
_LOGGER.debug(
|
|
|
|
"Sending SPC area command '%s' to area %s", command, area_id)
|
2017-06-05 06:53:25 +00:00
|
|
|
resource = "area/{}/{}".format(area_id, command)
|
|
|
|
return (yield from self._call_web_gateway(resource, use_get=False))
|
|
|
|
|
|
|
|
def start_listener(self, async_callback, *args):
|
|
|
|
"""Start the websocket listener."""
|
2018-03-12 18:42:08 +00:00
|
|
|
asyncio.ensure_future(self._ws_listen(async_callback, *args))
|
2017-06-05 06:53:25 +00:00
|
|
|
|
|
|
|
def _build_url(self, resource):
|
|
|
|
return urljoin(self._api_url, "spc/{}".format(resource))
|
|
|
|
|
|
|
|
@asyncio.coroutine
|
|
|
|
def _get_data(self, resource):
|
2018-01-21 06:35:38 +00:00
|
|
|
"""Get the data from the resource."""
|
2017-06-05 06:53:25 +00:00
|
|
|
data = yield from self._call_web_gateway(resource)
|
|
|
|
if not data:
|
|
|
|
return False
|
|
|
|
if data['status'] != 'success':
|
2018-01-21 06:35:38 +00:00
|
|
|
_LOGGER.error(
|
|
|
|
"SPC Web Gateway call unsuccessful for resource: %s", resource)
|
2017-06-05 06:53:25 +00:00
|
|
|
return False
|
|
|
|
return [item for item in data['data'][resource]]
|
|
|
|
|
|
|
|
@asyncio.coroutine
|
|
|
|
def _call_web_gateway(self, resource, use_get=True):
|
2018-01-21 06:35:38 +00:00
|
|
|
"""Call web gateway for data."""
|
2017-06-05 06:53:25 +00:00
|
|
|
response = None
|
|
|
|
session = None
|
|
|
|
url = self._build_url(resource)
|
|
|
|
try:
|
2018-01-21 06:35:38 +00:00
|
|
|
_LOGGER.debug("Attempting to retrieve SPC data from %s", url)
|
2018-03-04 05:28:04 +00:00
|
|
|
session = \
|
|
|
|
self._hass.helpers.aiohttp_client.async_get_clientsession()
|
2017-06-05 06:53:25 +00:00
|
|
|
with async_timeout.timeout(10, loop=self._hass.loop):
|
|
|
|
action = session.get if use_get else session.put
|
|
|
|
response = yield from action(url)
|
|
|
|
if response.status != 200:
|
2018-01-21 06:35:38 +00:00
|
|
|
_LOGGER.error(
|
|
|
|
"SPC Web Gateway returned http status %d, response %s",
|
|
|
|
response.status, (yield from response.text()))
|
2017-06-05 06:53:25 +00:00
|
|
|
return False
|
|
|
|
result = yield from response.json()
|
|
|
|
except asyncio.TimeoutError:
|
2018-01-21 06:35:38 +00:00
|
|
|
_LOGGER.error("Timeout getting SPC data from %s", url)
|
2017-06-05 06:53:25 +00:00
|
|
|
return False
|
|
|
|
except aiohttp.ClientError:
|
2018-01-21 06:35:38 +00:00
|
|
|
_LOGGER.exception("Error getting SPC data from %s", url)
|
2017-06-05 06:53:25 +00:00
|
|
|
return False
|
|
|
|
finally:
|
|
|
|
if session:
|
|
|
|
yield from session.close()
|
|
|
|
if response:
|
|
|
|
yield from response.release()
|
|
|
|
_LOGGER.debug("Data from SPC: %s", result)
|
|
|
|
return result
|
|
|
|
|
|
|
|
@asyncio.coroutine
|
|
|
|
def _ws_read(self):
|
2018-01-21 06:35:38 +00:00
|
|
|
"""Read from websocket."""
|
2017-06-05 06:53:25 +00:00
|
|
|
import websockets as wslib
|
|
|
|
|
|
|
|
try:
|
|
|
|
if not self._ws:
|
|
|
|
self._ws = yield from wslib.connect(self._ws_url)
|
2018-01-21 06:35:38 +00:00
|
|
|
_LOGGER.info("Connected to websocket at %s", self._ws_url)
|
2017-06-05 06:53:25 +00:00
|
|
|
except Exception as ws_exc: # pylint: disable=broad-except
|
|
|
|
_LOGGER.error("Failed to connect to websocket: %s", ws_exc)
|
|
|
|
return
|
|
|
|
|
|
|
|
result = None
|
|
|
|
|
|
|
|
try:
|
|
|
|
result = yield from self._ws.recv()
|
|
|
|
_LOGGER.debug("Data from websocket: %s", result)
|
|
|
|
except Exception as ws_exc: # pylint: disable=broad-except
|
|
|
|
_LOGGER.error("Failed to read from websocket: %s", ws_exc)
|
|
|
|
try:
|
|
|
|
yield from self._ws.close()
|
|
|
|
finally:
|
|
|
|
self._ws = None
|
|
|
|
|
|
|
|
return result
|
|
|
|
|
|
|
|
@asyncio.coroutine
|
|
|
|
def _ws_listen(self, async_callback, *args):
|
2018-01-21 06:35:38 +00:00
|
|
|
"""Listen on websocket."""
|
2017-06-05 06:53:25 +00:00
|
|
|
try:
|
|
|
|
while True:
|
|
|
|
result = yield from self._ws_read()
|
|
|
|
|
|
|
|
if result:
|
2018-01-21 06:35:38 +00:00
|
|
|
yield from _ws_process_message(
|
|
|
|
json.loads(result), async_callback, *args)
|
2017-06-05 06:53:25 +00:00
|
|
|
else:
|
2018-01-21 06:35:38 +00:00
|
|
|
_LOGGER.info("Trying again in 30 seconds")
|
2017-06-05 06:53:25 +00:00
|
|
|
yield from asyncio.sleep(30)
|
|
|
|
|
|
|
|
finally:
|
|
|
|
if self._ws:
|
|
|
|
yield from self._ws.close()
|