2019-02-13 20:21:14 +00:00
|
|
|
"""Support for SimpliSafe alarm control panels."""
|
2018-05-29 05:50:27 +00:00
|
|
|
import re
|
2016-07-02 18:21:15 +00:00
|
|
|
|
2020-02-13 19:30:38 +00:00
|
|
|
from simplipy.errors import SimplipyError
|
2019-07-30 23:23:42 +00:00
|
|
|
from simplipy.system import SystemStates
|
2020-02-13 19:30:38 +00:00
|
|
|
from simplipy.websocket import (
|
|
|
|
EVENT_ALARM_CANCELED,
|
|
|
|
EVENT_ALARM_TRIGGERED,
|
|
|
|
EVENT_ARMED_AWAY,
|
|
|
|
EVENT_ARMED_AWAY_BY_KEYPAD,
|
|
|
|
EVENT_ARMED_AWAY_BY_REMOTE,
|
|
|
|
EVENT_ARMED_HOME,
|
|
|
|
EVENT_AWAY_EXIT_DELAY_BY_KEYPAD,
|
|
|
|
EVENT_AWAY_EXIT_DELAY_BY_REMOTE,
|
|
|
|
EVENT_DISARMED_BY_MASTER_PIN,
|
|
|
|
EVENT_DISARMED_BY_REMOTE,
|
|
|
|
EVENT_HOME_EXIT_DELAY,
|
|
|
|
)
|
2019-07-30 23:23:42 +00:00
|
|
|
|
|
|
|
from homeassistant.components.alarm_control_panel import (
|
2019-07-31 19:25:30 +00:00
|
|
|
FORMAT_NUMBER,
|
|
|
|
FORMAT_TEXT,
|
2020-04-25 16:05:28 +00:00
|
|
|
AlarmControlPanelEntity,
|
2019-07-31 19:25:30 +00:00
|
|
|
)
|
2019-11-25 23:42:53 +00:00
|
|
|
from homeassistant.components.alarm_control_panel.const import (
|
|
|
|
SUPPORT_ALARM_ARM_AWAY,
|
|
|
|
SUPPORT_ALARM_ARM_HOME,
|
|
|
|
)
|
2016-07-02 18:21:15 +00:00
|
|
|
from homeassistant.const import (
|
2019-07-31 19:25:30 +00:00
|
|
|
CONF_CODE,
|
|
|
|
STATE_ALARM_ARMED_AWAY,
|
|
|
|
STATE_ALARM_ARMED_HOME,
|
2020-01-26 01:41:49 +00:00
|
|
|
STATE_ALARM_ARMING,
|
2019-07-31 19:25:30 +00:00
|
|
|
STATE_ALARM_DISARMED,
|
2020-01-26 01:41:49 +00:00
|
|
|
STATE_ALARM_TRIGGERED,
|
2019-07-31 19:25:30 +00:00
|
|
|
)
|
2020-02-13 19:30:38 +00:00
|
|
|
from homeassistant.core import callback
|
2016-07-02 18:21:15 +00:00
|
|
|
|
2019-11-09 22:41:02 +00:00
|
|
|
from . import SimpliSafeEntity
|
2020-02-13 19:30:38 +00:00
|
|
|
from .const import (
|
|
|
|
ATTR_ALARM_DURATION,
|
|
|
|
ATTR_ALARM_VOLUME,
|
|
|
|
ATTR_CHIME_VOLUME,
|
|
|
|
ATTR_ENTRY_DELAY_AWAY,
|
|
|
|
ATTR_ENTRY_DELAY_HOME,
|
|
|
|
ATTR_EXIT_DELAY_AWAY,
|
|
|
|
ATTR_EXIT_DELAY_HOME,
|
|
|
|
ATTR_LIGHT,
|
|
|
|
ATTR_VOICE_PROMPT_VOLUME,
|
|
|
|
DATA_CLIENT,
|
|
|
|
DOMAIN,
|
2020-10-17 19:40:34 +00:00
|
|
|
LOGGER,
|
2020-02-13 19:30:38 +00:00
|
|
|
VOLUME_STRING_MAP,
|
|
|
|
)
|
2019-03-21 05:56:46 +00:00
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
ATTR_BATTERY_BACKUP_POWER_LEVEL = "battery_backup_power_level"
|
|
|
|
ATTR_GSM_STRENGTH = "gsm_strength"
|
2020-02-13 19:30:38 +00:00
|
|
|
ATTR_PIN_NAME = "pin_name"
|
2019-07-31 19:25:30 +00:00
|
|
|
ATTR_RF_JAMMING = "rf_jamming"
|
|
|
|
ATTR_WALL_POWER_LEVEL = "wall_power_level"
|
|
|
|
ATTR_WIFI_STRENGTH = "wifi_strength"
|
|
|
|
|
|
|
|
|
2018-10-12 17:07:47 +00:00
|
|
|
async def async_setup_entry(hass, entry, async_add_entities):
|
|
|
|
"""Set up a SimpliSafe alarm control panel based on a config entry."""
|
2019-07-29 21:52:30 +00:00
|
|
|
simplisafe = hass.data[DOMAIN][DATA_CLIENT][entry.entry_id]
|
2019-07-31 19:25:30 +00:00
|
|
|
async_add_entities(
|
2020-03-13 05:00:00 +00:00
|
|
|
[SimpliSafeAlarm(simplisafe, system) for system in simplisafe.systems.values()],
|
2019-07-31 19:25:30 +00:00
|
|
|
True,
|
|
|
|
)
|
2018-07-19 17:13:46 +00:00
|
|
|
|
|
|
|
|
2020-04-25 16:05:28 +00:00
|
|
|
class SimpliSafeAlarm(SimpliSafeEntity, AlarmControlPanelEntity):
|
2018-01-21 06:35:38 +00:00
|
|
|
"""Representation of a SimpliSafe alarm."""
|
2016-07-02 18:21:15 +00:00
|
|
|
|
2020-03-13 05:00:00 +00:00
|
|
|
def __init__(self, simplisafe, system):
|
2016-07-12 14:46:29 +00:00
|
|
|
"""Initialize the SimpliSafe alarm."""
|
2020-02-13 19:30:38 +00:00
|
|
|
super().__init__(simplisafe, system, "Alarm Control Panel")
|
2019-07-30 23:23:42 +00:00
|
|
|
self._changed_by = None
|
2020-02-13 19:30:38 +00:00
|
|
|
self._last_event = None
|
2016-07-02 18:21:15 +00:00
|
|
|
|
2020-02-13 19:30:38 +00:00
|
|
|
if system.alarm_going_off:
|
|
|
|
self._state = STATE_ALARM_TRIGGERED
|
|
|
|
elif system.state == SystemStates.away:
|
|
|
|
self._state = STATE_ALARM_ARMED_AWAY
|
|
|
|
elif system.state in (
|
|
|
|
SystemStates.away_count,
|
|
|
|
SystemStates.exit_delay,
|
|
|
|
SystemStates.home_count,
|
|
|
|
):
|
|
|
|
self._state = STATE_ALARM_ARMING
|
|
|
|
elif system.state == SystemStates.home:
|
|
|
|
self._state = STATE_ALARM_ARMED_HOME
|
|
|
|
elif system.state == SystemStates.off:
|
|
|
|
self._state = STATE_ALARM_DISARMED
|
|
|
|
else:
|
|
|
|
self._state = None
|
|
|
|
|
|
|
|
for event_type in (
|
|
|
|
EVENT_ALARM_CANCELED,
|
|
|
|
EVENT_ALARM_TRIGGERED,
|
|
|
|
EVENT_ARMED_AWAY,
|
|
|
|
EVENT_ARMED_AWAY_BY_KEYPAD,
|
|
|
|
EVENT_ARMED_AWAY_BY_REMOTE,
|
|
|
|
EVENT_ARMED_HOME,
|
|
|
|
EVENT_AWAY_EXIT_DELAY_BY_KEYPAD,
|
|
|
|
EVENT_AWAY_EXIT_DELAY_BY_REMOTE,
|
|
|
|
EVENT_DISARMED_BY_MASTER_PIN,
|
|
|
|
EVENT_DISARMED_BY_REMOTE,
|
|
|
|
EVENT_HOME_EXIT_DELAY,
|
|
|
|
):
|
|
|
|
self.websocket_events_to_listen_for.append(event_type)
|
2016-07-02 18:21:15 +00:00
|
|
|
|
2019-07-30 23:23:42 +00:00
|
|
|
@property
|
|
|
|
def changed_by(self):
|
|
|
|
"""Return info about who changed the alarm last."""
|
|
|
|
return self._changed_by
|
|
|
|
|
2016-07-02 18:21:15 +00:00
|
|
|
@property
|
|
|
|
def code_format(self):
|
2018-05-29 05:50:27 +00:00
|
|
|
"""Return one or more digits/characters."""
|
2020-10-15 16:36:04 +00:00
|
|
|
if not self._simplisafe.config_entry.options.get(CONF_CODE):
|
2018-05-29 05:50:27 +00:00
|
|
|
return None
|
2020-10-15 16:36:04 +00:00
|
|
|
if isinstance(
|
|
|
|
self._simplisafe.config_entry.options[CONF_CODE], str
|
|
|
|
) and re.search("^\\d+$", self._simplisafe.config_entry.options[CONF_CODE]):
|
2019-07-30 23:23:42 +00:00
|
|
|
return FORMAT_NUMBER
|
|
|
|
return FORMAT_TEXT
|
2016-07-02 18:21:15 +00:00
|
|
|
|
2019-07-17 22:13:03 +00:00
|
|
|
@property
|
|
|
|
def state(self):
|
2019-07-30 23:23:42 +00:00
|
|
|
"""Return the state of the entity."""
|
2019-07-17 22:13:03 +00:00
|
|
|
return self._state
|
|
|
|
|
2019-11-25 23:42:53 +00:00
|
|
|
@property
|
|
|
|
def supported_features(self) -> int:
|
|
|
|
"""Return the list of supported features."""
|
|
|
|
return SUPPORT_ALARM_ARM_HOME | SUPPORT_ALARM_ARM_AWAY
|
|
|
|
|
2020-03-13 05:00:00 +00:00
|
|
|
@callback
|
|
|
|
def _is_code_valid(self, code, state):
|
|
|
|
"""Validate that a code matches the required one."""
|
2020-10-15 16:36:04 +00:00
|
|
|
if not self._simplisafe.config_entry.options.get(CONF_CODE):
|
2020-03-13 05:00:00 +00:00
|
|
|
return True
|
|
|
|
|
2020-10-15 16:36:04 +00:00
|
|
|
if not code or code != self._simplisafe.config_entry.options[CONF_CODE]:
|
2020-10-17 19:40:34 +00:00
|
|
|
LOGGER.warning(
|
2020-03-13 05:00:00 +00:00
|
|
|
"Incorrect alarm code entered (target state: %s): %s", state, code
|
|
|
|
)
|
|
|
|
return False
|
|
|
|
|
|
|
|
return True
|
2016-07-02 18:21:15 +00:00
|
|
|
|
2018-10-04 03:03:29 +00:00
|
|
|
async def async_alarm_disarm(self, code=None):
|
2016-07-02 18:21:15 +00:00
|
|
|
"""Send disarm command."""
|
2020-03-13 05:00:00 +00:00
|
|
|
if not self._is_code_valid(code, STATE_ALARM_DISARMED):
|
2016-07-07 04:55:47 +00:00
|
|
|
return
|
2016-07-02 18:21:15 +00:00
|
|
|
|
2020-02-13 19:30:38 +00:00
|
|
|
try:
|
|
|
|
await self._system.set_off()
|
|
|
|
except SimplipyError as err:
|
2020-10-17 19:40:34 +00:00
|
|
|
LOGGER.error('Error while disarming "%s": %s', self._system.name, err)
|
2020-02-13 19:30:38 +00:00
|
|
|
return
|
|
|
|
|
|
|
|
self._state = STATE_ALARM_DISARMED
|
2018-10-04 03:03:29 +00:00
|
|
|
|
|
|
|
async def async_alarm_arm_home(self, code=None):
|
2016-07-02 18:21:15 +00:00
|
|
|
"""Send arm home command."""
|
2020-03-13 05:00:00 +00:00
|
|
|
if not self._is_code_valid(code, STATE_ALARM_ARMED_HOME):
|
2016-07-07 04:55:47 +00:00
|
|
|
return
|
2016-07-02 18:21:15 +00:00
|
|
|
|
2020-02-13 19:30:38 +00:00
|
|
|
try:
|
|
|
|
await self._system.set_home()
|
|
|
|
except SimplipyError as err:
|
2020-10-17 19:40:34 +00:00
|
|
|
LOGGER.error('Error while arming "%s" (home): %s', self._system.name, err)
|
2020-02-13 19:30:38 +00:00
|
|
|
return
|
|
|
|
|
|
|
|
self._state = STATE_ALARM_ARMED_HOME
|
2018-10-04 03:03:29 +00:00
|
|
|
|
|
|
|
async def async_alarm_arm_away(self, code=None):
|
2016-07-02 18:21:15 +00:00
|
|
|
"""Send arm away command."""
|
2020-03-13 05:00:00 +00:00
|
|
|
if not self._is_code_valid(code, STATE_ALARM_ARMED_AWAY):
|
2016-07-07 04:55:47 +00:00
|
|
|
return
|
|
|
|
|
2020-02-13 19:30:38 +00:00
|
|
|
try:
|
|
|
|
await self._system.set_away()
|
|
|
|
except SimplipyError as err:
|
2020-10-17 19:40:34 +00:00
|
|
|
LOGGER.error('Error while arming "%s" (away): %s', self._system.name, err)
|
2020-02-13 19:30:38 +00:00
|
|
|
return
|
2018-10-12 17:07:47 +00:00
|
|
|
|
2020-02-13 19:30:38 +00:00
|
|
|
self._state = STATE_ALARM_ARMING
|
2018-10-12 17:07:47 +00:00
|
|
|
|
2020-02-13 19:30:38 +00:00
|
|
|
@callback
|
|
|
|
def async_update_from_rest_api(self):
|
|
|
|
"""Update the entity with the provided REST API data."""
|
|
|
|
if self._system.version == 3:
|
|
|
|
self._attrs.update(
|
|
|
|
{
|
|
|
|
ATTR_ALARM_DURATION: self._system.alarm_duration,
|
|
|
|
ATTR_ALARM_VOLUME: VOLUME_STRING_MAP[self._system.alarm_volume],
|
|
|
|
ATTR_BATTERY_BACKUP_POWER_LEVEL: self._system.battery_backup_power_level,
|
|
|
|
ATTR_CHIME_VOLUME: VOLUME_STRING_MAP[self._system.chime_volume],
|
|
|
|
ATTR_ENTRY_DELAY_AWAY: self._system.entry_delay_away,
|
|
|
|
ATTR_ENTRY_DELAY_HOME: self._system.entry_delay_home,
|
|
|
|
ATTR_EXIT_DELAY_AWAY: self._system.exit_delay_away,
|
|
|
|
ATTR_EXIT_DELAY_HOME: self._system.exit_delay_home,
|
|
|
|
ATTR_GSM_STRENGTH: self._system.gsm_strength,
|
|
|
|
ATTR_LIGHT: self._system.light,
|
|
|
|
ATTR_RF_JAMMING: self._system.rf_jamming,
|
|
|
|
ATTR_VOICE_PROMPT_VOLUME: VOLUME_STRING_MAP[
|
|
|
|
self._system.voice_prompt_volume
|
|
|
|
],
|
|
|
|
ATTR_WALL_POWER_LEVEL: self._system.wall_power_level,
|
|
|
|
ATTR_WIFI_STRENGTH: self._system.wifi_strength,
|
|
|
|
}
|
|
|
|
)
|
|
|
|
|
2020-04-05 16:47:04 +00:00
|
|
|
# Although system state updates are designed the come via the websocket, the
|
|
|
|
# SimpliSafe cloud can sporadically fail to send those updates as expected; so,
|
|
|
|
# just in case, we synchronize the state via the REST API, too:
|
2020-04-14 17:53:53 +00:00
|
|
|
if self._system.state == SystemStates.alarm:
|
|
|
|
self._state = STATE_ALARM_TRIGGERED
|
|
|
|
elif self._system.state == SystemStates.away:
|
2020-04-05 16:47:04 +00:00
|
|
|
self._state = STATE_ALARM_ARMED_AWAY
|
|
|
|
elif self._system.state in (SystemStates.away_count, SystemStates.exit_delay):
|
|
|
|
self._state = STATE_ALARM_ARMING
|
|
|
|
elif self._system.state == SystemStates.home:
|
|
|
|
self._state = STATE_ALARM_ARMED_HOME
|
|
|
|
elif self._system.state == SystemStates.off:
|
|
|
|
self._state = STATE_ALARM_DISARMED
|
|
|
|
else:
|
|
|
|
self._state = None
|
|
|
|
|
2020-02-13 19:30:38 +00:00
|
|
|
@callback
|
|
|
|
def async_update_from_websocket_event(self, event):
|
|
|
|
"""Update the entity with the provided websocket API event data."""
|
|
|
|
if event.event_type in (
|
|
|
|
EVENT_ALARM_CANCELED,
|
|
|
|
EVENT_DISARMED_BY_MASTER_PIN,
|
|
|
|
EVENT_DISARMED_BY_REMOTE,
|
|
|
|
):
|
|
|
|
self._state = STATE_ALARM_DISARMED
|
|
|
|
elif event.event_type == EVENT_ALARM_TRIGGERED:
|
2020-01-26 01:41:49 +00:00
|
|
|
self._state = STATE_ALARM_TRIGGERED
|
2020-02-13 19:30:38 +00:00
|
|
|
elif event.event_type in (
|
|
|
|
EVENT_ARMED_AWAY,
|
|
|
|
EVENT_ARMED_AWAY_BY_KEYPAD,
|
|
|
|
EVENT_ARMED_AWAY_BY_REMOTE,
|
|
|
|
):
|
2020-01-26 01:41:49 +00:00
|
|
|
self._state = STATE_ALARM_ARMED_AWAY
|
2020-02-13 19:30:38 +00:00
|
|
|
elif event.event_type == EVENT_ARMED_HOME:
|
|
|
|
self._state = STATE_ALARM_ARMED_HOME
|
|
|
|
elif event.event_type in (
|
|
|
|
EVENT_AWAY_EXIT_DELAY_BY_KEYPAD,
|
|
|
|
EVENT_AWAY_EXIT_DELAY_BY_REMOTE,
|
|
|
|
EVENT_HOME_EXIT_DELAY,
|
2019-07-31 19:25:30 +00:00
|
|
|
):
|
2020-01-26 01:41:49 +00:00
|
|
|
self._state = STATE_ALARM_ARMING
|
2018-10-04 03:03:29 +00:00
|
|
|
else:
|
|
|
|
self._state = None
|
2019-07-30 23:23:42 +00:00
|
|
|
|
2020-02-13 19:30:38 +00:00
|
|
|
self._changed_by = event.changed_by
|