2019-02-13 20:21:14 +00:00
|
|
|
"""Support for Envisalink devices."""
|
2017-02-23 21:02:56 +00:00
|
|
|
import asyncio
|
2016-06-19 17:45:07 +00:00
|
|
|
import logging
|
2017-02-23 21:02:56 +00:00
|
|
|
|
2016-06-19 17:45:07 +00:00
|
|
|
import voluptuous as vol
|
2017-02-23 21:02:56 +00:00
|
|
|
|
|
|
|
from homeassistant.core import callback
|
2016-06-19 17:45:07 +00:00
|
|
|
import homeassistant.helpers.config_validation as cv
|
2019-07-31 19:25:30 +00:00
|
|
|
from homeassistant.const import EVENT_HOMEASSISTANT_STOP, CONF_TIMEOUT, CONF_HOST
|
2016-06-19 17:45:07 +00:00
|
|
|
from homeassistant.helpers.entity import Entity
|
2017-02-23 21:02:56 +00:00
|
|
|
from homeassistant.helpers.discovery import async_load_platform
|
|
|
|
from homeassistant.helpers.dispatcher import async_dispatcher_send
|
2016-06-19 17:45:07 +00:00
|
|
|
|
|
|
|
_LOGGER = logging.getLogger(__name__)
|
2017-04-30 05:04:49 +00:00
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
DOMAIN = "envisalink"
|
|
|
|
|
|
|
|
DATA_EVL = "envisalink"
|
|
|
|
|
|
|
|
CONF_CODE = "code"
|
|
|
|
CONF_EVL_KEEPALIVE = "keepalive_interval"
|
|
|
|
CONF_EVL_PORT = "port"
|
|
|
|
CONF_EVL_VERSION = "evl_version"
|
|
|
|
CONF_PANEL_TYPE = "panel_type"
|
|
|
|
CONF_PANIC = "panic_type"
|
|
|
|
CONF_PARTITIONNAME = "name"
|
|
|
|
CONF_PARTITIONS = "partitions"
|
|
|
|
CONF_PASS = "password"
|
|
|
|
CONF_USERNAME = "user_name"
|
|
|
|
CONF_ZONEDUMP_INTERVAL = "zonedump_interval"
|
|
|
|
CONF_ZONENAME = "name"
|
|
|
|
CONF_ZONES = "zones"
|
|
|
|
CONF_ZONETYPE = "type"
|
2016-06-19 17:45:07 +00:00
|
|
|
|
|
|
|
DEFAULT_PORT = 4025
|
|
|
|
DEFAULT_EVL_VERSION = 3
|
|
|
|
DEFAULT_KEEPALIVE = 60
|
|
|
|
DEFAULT_ZONEDUMP_INTERVAL = 30
|
2019-07-31 19:25:30 +00:00
|
|
|
DEFAULT_ZONETYPE = "opening"
|
|
|
|
DEFAULT_PANIC = "Police"
|
2019-01-02 12:46:33 +00:00
|
|
|
DEFAULT_TIMEOUT = 10
|
2016-06-19 17:45:07 +00:00
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
SIGNAL_ZONE_UPDATE = "envisalink.zones_updated"
|
|
|
|
SIGNAL_PARTITION_UPDATE = "envisalink.partition_updated"
|
|
|
|
SIGNAL_KEYPAD_UPDATE = "envisalink.keypad_updated"
|
|
|
|
|
|
|
|
ZONE_SCHEMA = vol.Schema(
|
|
|
|
{
|
|
|
|
vol.Required(CONF_ZONENAME): cv.string,
|
|
|
|
vol.Optional(CONF_ZONETYPE, default=DEFAULT_ZONETYPE): cv.string,
|
|
|
|
}
|
|
|
|
)
|
|
|
|
|
|
|
|
PARTITION_SCHEMA = vol.Schema({vol.Required(CONF_PARTITIONNAME): cv.string})
|
|
|
|
|
|
|
|
CONFIG_SCHEMA = vol.Schema(
|
|
|
|
{
|
|
|
|
DOMAIN: vol.Schema(
|
|
|
|
{
|
|
|
|
vol.Required(CONF_HOST): cv.string,
|
|
|
|
vol.Required(CONF_PANEL_TYPE): vol.All(
|
|
|
|
cv.string, vol.In(["HONEYWELL", "DSC"])
|
|
|
|
),
|
|
|
|
vol.Required(CONF_USERNAME): cv.string,
|
|
|
|
vol.Required(CONF_PASS): cv.string,
|
|
|
|
vol.Optional(CONF_CODE): cv.string,
|
|
|
|
vol.Optional(CONF_PANIC, default=DEFAULT_PANIC): cv.string,
|
|
|
|
vol.Optional(CONF_ZONES): {vol.Coerce(int): ZONE_SCHEMA},
|
|
|
|
vol.Optional(CONF_PARTITIONS): {vol.Coerce(int): PARTITION_SCHEMA},
|
|
|
|
vol.Optional(CONF_EVL_PORT, default=DEFAULT_PORT): cv.port,
|
|
|
|
vol.Optional(CONF_EVL_VERSION, default=DEFAULT_EVL_VERSION): vol.All(
|
|
|
|
vol.Coerce(int), vol.Range(min=3, max=4)
|
|
|
|
),
|
|
|
|
vol.Optional(CONF_EVL_KEEPALIVE, default=DEFAULT_KEEPALIVE): vol.All(
|
|
|
|
vol.Coerce(int), vol.Range(min=15)
|
|
|
|
),
|
|
|
|
vol.Optional(
|
|
|
|
CONF_ZONEDUMP_INTERVAL, default=DEFAULT_ZONEDUMP_INTERVAL
|
|
|
|
): vol.Coerce(int),
|
|
|
|
vol.Optional(CONF_TIMEOUT, default=DEFAULT_TIMEOUT): vol.Coerce(int),
|
|
|
|
}
|
|
|
|
)
|
|
|
|
},
|
|
|
|
extra=vol.ALLOW_EXTRA,
|
|
|
|
)
|
|
|
|
|
|
|
|
SERVICE_CUSTOM_FUNCTION = "invoke_custom_function"
|
|
|
|
ATTR_CUSTOM_FUNCTION = "pgm"
|
|
|
|
ATTR_PARTITION = "partition"
|
|
|
|
|
|
|
|
SERVICE_SCHEMA = vol.Schema(
|
|
|
|
{
|
|
|
|
vol.Required(ATTR_CUSTOM_FUNCTION): cv.string,
|
|
|
|
vol.Required(ATTR_PARTITION): cv.string,
|
|
|
|
}
|
|
|
|
)
|
2019-01-02 12:46:33 +00:00
|
|
|
|
2016-06-19 17:45:07 +00:00
|
|
|
|
2018-10-01 06:52:42 +00:00
|
|
|
async def async_setup(hass, config):
|
2017-04-30 05:04:49 +00:00
|
|
|
"""Set up for Envisalink devices."""
|
2016-06-19 17:45:07 +00:00
|
|
|
from pyenvisalink import EnvisalinkAlarmPanel
|
|
|
|
|
2017-02-23 21:02:56 +00:00
|
|
|
conf = config.get(DOMAIN)
|
|
|
|
|
2019-03-28 02:48:05 +00:00
|
|
|
host = conf.get(CONF_HOST)
|
2017-02-23 21:02:56 +00:00
|
|
|
port = conf.get(CONF_EVL_PORT)
|
|
|
|
code = conf.get(CONF_CODE)
|
|
|
|
panel_type = conf.get(CONF_PANEL_TYPE)
|
|
|
|
panic_type = conf.get(CONF_PANIC)
|
|
|
|
version = conf.get(CONF_EVL_VERSION)
|
|
|
|
user = conf.get(CONF_USERNAME)
|
|
|
|
password = conf.get(CONF_PASS)
|
|
|
|
keep_alive = conf.get(CONF_EVL_KEEPALIVE)
|
|
|
|
zone_dump = conf.get(CONF_ZONEDUMP_INTERVAL)
|
|
|
|
zones = conf.get(CONF_ZONES)
|
|
|
|
partitions = conf.get(CONF_PARTITIONS)
|
2019-01-02 12:46:33 +00:00
|
|
|
connection_timeout = conf.get(CONF_TIMEOUT)
|
2019-05-23 04:09:59 +00:00
|
|
|
sync_connect = asyncio.Future()
|
2017-02-23 21:02:56 +00:00
|
|
|
|
|
|
|
controller = EnvisalinkAlarmPanel(
|
2019-07-31 19:25:30 +00:00
|
|
|
host,
|
|
|
|
port,
|
|
|
|
panel_type,
|
|
|
|
version,
|
|
|
|
user,
|
|
|
|
password,
|
|
|
|
zone_dump,
|
|
|
|
keep_alive,
|
|
|
|
hass.loop,
|
|
|
|
connection_timeout,
|
|
|
|
)
|
2017-02-23 21:02:56 +00:00
|
|
|
hass.data[DATA_EVL] = controller
|
|
|
|
|
|
|
|
@callback
|
2016-06-19 17:45:07 +00:00
|
|
|
def login_fail_callback(data):
|
2017-05-02 20:47:20 +00:00
|
|
|
"""Handle when the evl rejects our login."""
|
2017-04-30 05:04:49 +00:00
|
|
|
_LOGGER.error("The Envisalink rejected your credentials")
|
2018-08-05 16:51:23 +00:00
|
|
|
if not sync_connect.done():
|
|
|
|
sync_connect.set_result(False)
|
2016-06-19 17:45:07 +00:00
|
|
|
|
2017-02-23 21:02:56 +00:00
|
|
|
@callback
|
2016-06-19 17:45:07 +00:00
|
|
|
def connection_fail_callback(data):
|
|
|
|
"""Network failure callback."""
|
2019-10-04 00:15:52 +00:00
|
|
|
_LOGGER.error(
|
|
|
|
"Could not establish a connection with the Envisalink- retrying..."
|
|
|
|
)
|
2018-08-05 16:51:23 +00:00
|
|
|
if not sync_connect.done():
|
2019-10-04 00:15:52 +00:00
|
|
|
hass.bus.async_listen_once(EVENT_HOMEASSISTANT_STOP, stop_envisalink)
|
|
|
|
sync_connect.set_result(True)
|
2016-06-19 17:45:07 +00:00
|
|
|
|
2017-02-23 21:02:56 +00:00
|
|
|
@callback
|
2016-06-19 17:45:07 +00:00
|
|
|
def connection_success_callback(data):
|
2017-05-02 20:47:20 +00:00
|
|
|
"""Handle a successful connection."""
|
2017-04-30 05:04:49 +00:00
|
|
|
_LOGGER.info("Established a connection with the Envisalink")
|
2018-08-05 16:51:23 +00:00
|
|
|
if not sync_connect.done():
|
2019-07-31 19:25:30 +00:00
|
|
|
hass.bus.async_listen_once(EVENT_HOMEASSISTANT_STOP, stop_envisalink)
|
2018-08-05 16:51:23 +00:00
|
|
|
sync_connect.set_result(True)
|
2016-06-19 17:45:07 +00:00
|
|
|
|
2017-02-23 21:02:56 +00:00
|
|
|
@callback
|
2016-06-19 17:45:07 +00:00
|
|
|
def zones_updated_callback(data):
|
|
|
|
"""Handle zone timer updates."""
|
2019-01-21 00:32:01 +00:00
|
|
|
_LOGGER.debug("Envisalink sent a zone update event. Updating zones...")
|
2017-02-23 21:02:56 +00:00
|
|
|
async_dispatcher_send(hass, SIGNAL_ZONE_UPDATE, data)
|
2016-06-19 17:45:07 +00:00
|
|
|
|
2017-02-23 21:02:56 +00:00
|
|
|
@callback
|
2016-06-19 17:45:07 +00:00
|
|
|
def alarm_data_updated_callback(data):
|
|
|
|
"""Handle non-alarm based info updates."""
|
2019-01-21 00:32:01 +00:00
|
|
|
_LOGGER.debug("Envisalink sent new alarm info. Updating alarms...")
|
2017-02-23 21:02:56 +00:00
|
|
|
async_dispatcher_send(hass, SIGNAL_KEYPAD_UPDATE, data)
|
2016-06-19 17:45:07 +00:00
|
|
|
|
2017-02-23 21:02:56 +00:00
|
|
|
@callback
|
2016-06-19 17:45:07 +00:00
|
|
|
def partition_updated_callback(data):
|
|
|
|
"""Handle partition changes thrown by evl (including alarms)."""
|
2019-01-21 00:32:01 +00:00
|
|
|
_LOGGER.debug("The envisalink sent a partition update event")
|
2017-02-23 21:02:56 +00:00
|
|
|
async_dispatcher_send(hass, SIGNAL_PARTITION_UPDATE, data)
|
2016-06-19 17:45:07 +00:00
|
|
|
|
2017-02-23 21:02:56 +00:00
|
|
|
@callback
|
2016-06-19 17:45:07 +00:00
|
|
|
def stop_envisalink(event):
|
|
|
|
"""Shutdown envisalink connection and thread on exit."""
|
2017-04-30 05:04:49 +00:00
|
|
|
_LOGGER.info("Shutting down Envisalink")
|
2017-02-23 21:02:56 +00:00
|
|
|
controller.stop()
|
|
|
|
|
2019-01-02 12:46:33 +00:00
|
|
|
async def handle_custom_function(call):
|
|
|
|
"""Handle custom/PGM service."""
|
|
|
|
custom_function = call.data.get(ATTR_CUSTOM_FUNCTION)
|
|
|
|
partition = call.data.get(ATTR_PARTITION)
|
|
|
|
controller.command_output(code, partition, custom_function)
|
|
|
|
|
2017-02-23 21:02:56 +00:00
|
|
|
controller.callback_zone_timer_dump = zones_updated_callback
|
|
|
|
controller.callback_zone_state_change = zones_updated_callback
|
|
|
|
controller.callback_partition_state_change = partition_updated_callback
|
|
|
|
controller.callback_keypad_update = alarm_data_updated_callback
|
|
|
|
controller.callback_login_failure = login_fail_callback
|
|
|
|
controller.callback_login_timeout = connection_fail_callback
|
|
|
|
controller.callback_login_success = connection_success_callback
|
2016-06-19 17:45:07 +00:00
|
|
|
|
2017-02-23 21:02:56 +00:00
|
|
|
_LOGGER.info("Start envisalink.")
|
|
|
|
controller.start()
|
2016-06-19 17:45:07 +00:00
|
|
|
|
2018-10-01 06:52:42 +00:00
|
|
|
result = await sync_connect
|
2017-02-23 21:02:56 +00:00
|
|
|
if not result:
|
2016-06-19 17:45:07 +00:00
|
|
|
return False
|
|
|
|
|
2016-07-01 19:39:30 +00:00
|
|
|
# Load sub-components for Envisalink
|
2017-02-23 21:02:56 +00:00
|
|
|
if partitions:
|
2019-07-31 19:25:30 +00:00
|
|
|
hass.async_create_task(
|
|
|
|
async_load_platform(
|
|
|
|
hass,
|
|
|
|
"alarm_control_panel",
|
|
|
|
"envisalink",
|
|
|
|
{CONF_PARTITIONS: partitions, CONF_CODE: code, CONF_PANIC: panic_type},
|
|
|
|
config,
|
|
|
|
)
|
|
|
|
)
|
|
|
|
hass.async_create_task(
|
|
|
|
async_load_platform(
|
|
|
|
hass,
|
|
|
|
"sensor",
|
|
|
|
"envisalink",
|
|
|
|
{CONF_PARTITIONS: partitions, CONF_CODE: code},
|
|
|
|
config,
|
|
|
|
)
|
|
|
|
)
|
2017-02-23 21:02:56 +00:00
|
|
|
if zones:
|
2019-07-31 19:25:30 +00:00
|
|
|
hass.async_create_task(
|
|
|
|
async_load_platform(
|
|
|
|
hass, "binary_sensor", "envisalink", {CONF_ZONES: zones}, config
|
|
|
|
)
|
|
|
|
)
|
|
|
|
|
|
|
|
hass.services.async_register(
|
|
|
|
DOMAIN, SERVICE_CUSTOM_FUNCTION, handle_custom_function, schema=SERVICE_SCHEMA
|
|
|
|
)
|
2019-01-02 12:46:33 +00:00
|
|
|
|
2016-06-19 17:45:07 +00:00
|
|
|
return True
|
|
|
|
|
|
|
|
|
|
|
|
class EnvisalinkDevice(Entity):
|
2016-07-01 19:39:30 +00:00
|
|
|
"""Representation of an Envisalink device."""
|
2016-06-19 17:45:07 +00:00
|
|
|
|
|
|
|
def __init__(self, name, info, controller):
|
|
|
|
"""Initialize the device."""
|
|
|
|
self._controller = controller
|
|
|
|
self._info = info
|
|
|
|
self._name = name
|
|
|
|
|
|
|
|
@property
|
|
|
|
def name(self):
|
|
|
|
"""Return the name of the device."""
|
|
|
|
return self._name
|
|
|
|
|
|
|
|
@property
|
|
|
|
def should_poll(self):
|
|
|
|
"""No polling needed."""
|
|
|
|
return False
|