Add totalconnect zones as binary sensors (#28712)

* Bump skybellpy to 0.4.0

* Bump skybellpy to 0.4.0 in requirements_all.txt

* Added extra states for STATE_ALARM_TRIGGERED to allow users to know if
it is a burglar or fire or carbon monoxide so automations can take
appropriate actions.  Updated TotalConnect component to handle these new
states.

* Fix const import

* Fix const import

* Fix const imports

* Bump total-connect-client to 0.26.

* Catch details of alarm trigger in state attributes.

Also bumps total_connect_client to 0.27.

* Change state_attributes() to device_state_attributes()

* Move totalconnect component toward being a multi-platform integration.  Bump total_connect_client to 0.28.

* add missing total-connect alarm state mappings

* Made recommended changes of MartinHjelmare at
https://github.com/home-assistant/home-assistant/pull/24427

* Update __init__.py

* Updates per MartinHjelmare comments

* flake8/pydocstyle fixes

* removed . at end of log message

* added blank line between logging and voluptuous

* more fixes

* Adding totalconnect zones as HA binary_sensors

* fix manifest.json

* flake8/pydocstyle fixes.  Added codeowner.

* Update formatting per @springstan guidance.

* Fixed pylint

* Add zone ID to log message for easier troubleshooting

* Account for bypassed zones in update()

* More status handling fixes.

* Fixed flake8 error

* Another attempt at black/isort fixes.

* Bump total-connect-client to 0.50.  Simplify code using new functions in
total-connect-client package instead of importing constants.  Run black
and isort.

* Fix manifest file

* Another manifest fix

* one more manifest fix

* more manifest changes.

* sync up

* fix indent

* one more pylint fix

* Hopefully the last pylint fix

* make variable names understandable

* create and fill dict in one step

* Fix name and attributes

* rename to logical variable in alarm_control_panel

* Remove location_name from alarm_control_panel attributes since it is
already the name of the alarm.

* Multiple fixes to improve code per @springstan suggestions

* Update homeassistant/components/totalconnect/binary_sensor.py

Co-Authored-By: springstan <46536646+springstan@users.noreply.github.com>

* Multiple changes per @MartinHjelmare review

* simplify alarm adding

* Fix binary_sensor.py is_on

Co-authored-by: springstan <46536646+springstan@users.noreply.github.com>
This commit is contained in:
Austin Mroczek 2020-02-01 01:09:52 -08:00 committed by GitHub
parent 26415f6abd
commit f584df46b7
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23
6 changed files with 120 additions and 30 deletions

View file

@ -354,6 +354,7 @@ homeassistant/components/time_date/* @fabaff
homeassistant/components/tmb/* @alemuro homeassistant/components/tmb/* @alemuro
homeassistant/components/todoist/* @boralyl homeassistant/components/todoist/* @boralyl
homeassistant/components/toon/* @frenck homeassistant/components/toon/* @frenck
homeassistant/components/totalconnect/* @austinmroczek
homeassistant/components/tplink/* @rytilahti homeassistant/components/tplink/* @rytilahti
homeassistant/components/traccar/* @ludeeus homeassistant/components/traccar/* @ludeeus
homeassistant/components/tradfri/* @ggravlingen homeassistant/components/tradfri/* @ggravlingen

View file

@ -24,7 +24,7 @@ CONFIG_SCHEMA = vol.Schema(
extra=vol.ALLOW_EXTRA, extra=vol.ALLOW_EXTRA,
) )
TOTALCONNECT_PLATFORMS = ["alarm_control_panel"] TOTALCONNECT_PLATFORMS = ["alarm_control_panel", "binary_sensor"]
def setup(hass, config): def setup(hass, config):

View file

@ -32,10 +32,9 @@ def setup_platform(hass, config, add_entities, discovery_info=None):
client = hass.data[TOTALCONNECT_DOMAIN].client client = hass.data[TOTALCONNECT_DOMAIN].client
for location in client.locations: for location_id, location in client.locations.items():
location_id = location.get("LocationID") location_name = location.location_name
name = location.get("LocationName") alarms.append(TotalConnectAlarm(location_name, location_id, client))
alarms.append(TotalConnectAlarm(name, location_id, client))
add_entities(alarms) add_entities(alarms)
@ -72,35 +71,35 @@ class TotalConnectAlarm(alarm.AlarmControlPanel):
def update(self): def update(self):
"""Return the state of the device.""" """Return the state of the device."""
status = self._client.get_armed_status(self._name) status = self._client.get_armed_status(self._location_id)
attr = { attr = {
"location_name": self._name, "location_name": self._name,
"location_id": self._location_id, "location_id": self._location_id,
"ac_loss": self._client.ac_loss, "ac_loss": self._client.locations[self._location_id].ac_loss,
"low_battery": self._client.low_battery, "low_battery": self._client.locations[self._location_id].low_battery,
"cover_tampered": self._client.locations[
self._location_id
].is_cover_tampered,
"triggered_source": None, "triggered_source": None,
"triggered_zone": None, "triggered_zone": None,
} }
if status == self._client.DISARMED: if status in (self._client.DISARMED, self._client.DISARMED_BYPASS):
state = STATE_ALARM_DISARMED state = STATE_ALARM_DISARMED
elif status == self._client.DISARMED_BYPASS: elif status in (
state = STATE_ALARM_DISARMED self._client.ARMED_STAY,
elif status == self._client.ARMED_STAY: self._client.ARMED_STAY_INSTANT,
state = STATE_ALARM_ARMED_HOME self._client.ARMED_STAY_INSTANT_BYPASS,
elif status == self._client.ARMED_STAY_INSTANT: ):
state = STATE_ALARM_ARMED_HOME
elif status == self._client.ARMED_STAY_INSTANT_BYPASS:
state = STATE_ALARM_ARMED_HOME state = STATE_ALARM_ARMED_HOME
elif status == self._client.ARMED_STAY_NIGHT: elif status == self._client.ARMED_STAY_NIGHT:
state = STATE_ALARM_ARMED_NIGHT state = STATE_ALARM_ARMED_NIGHT
elif status == self._client.ARMED_AWAY: elif status in (
state = STATE_ALARM_ARMED_AWAY self._client.ARMED_AWAY,
elif status == self._client.ARMED_AWAY_BYPASS: self._client.ARMED_AWAY_BYPASS,
state = STATE_ALARM_ARMED_AWAY self._client.ARMED_AWAY_INSTANT,
elif status == self._client.ARMED_AWAY_INSTANT: self._client.ARMED_AWAY_INSTANT_BYPASS,
state = STATE_ALARM_ARMED_AWAY ):
elif status == self._client.ARMED_AWAY_INSTANT_BYPASS:
state = STATE_ALARM_ARMED_AWAY state = STATE_ALARM_ARMED_AWAY
elif status == self._client.ARMED_CUSTOM_BYPASS: elif status == self._client.ARMED_CUSTOM_BYPASS:
state = STATE_ALARM_ARMED_CUSTOM_BYPASS state = STATE_ALARM_ARMED_CUSTOM_BYPASS
@ -128,16 +127,16 @@ class TotalConnectAlarm(alarm.AlarmControlPanel):
def alarm_disarm(self, code=None): def alarm_disarm(self, code=None):
"""Send disarm command.""" """Send disarm command."""
self._client.disarm(self._name) self._client.disarm(self._location_id)
def alarm_arm_home(self, code=None): def alarm_arm_home(self, code=None):
"""Send arm home command.""" """Send arm home command."""
self._client.arm_stay(self._name) self._client.arm_stay(self._location_id)
def alarm_arm_away(self, code=None): def alarm_arm_away(self, code=None):
"""Send arm away command.""" """Send arm away command."""
self._client.arm_away(self._name) self._client.arm_away(self._location_id)
def alarm_arm_night(self, code=None): def alarm_arm_night(self, code=None):
"""Send arm night command.""" """Send arm night command."""
self._client.arm_stay_night(self._name) self._client.arm_stay_night(self._location_id)

View file

@ -0,0 +1,90 @@
"""Interfaces with TotalConnect sensors."""
import logging
from homeassistant.components.binary_sensor import (
DEVICE_CLASS_DOOR,
DEVICE_CLASS_GAS,
DEVICE_CLASS_SMOKE,
BinarySensorDevice,
)
from . import DOMAIN as TOTALCONNECT_DOMAIN
_LOGGER = logging.getLogger(__name__)
def setup_platform(hass, config, add_entities, discovery_info=None):
"""Set up a sensor for a TotalConnect device."""
if discovery_info is None:
return
sensors = []
client_locations = hass.data[TOTALCONNECT_DOMAIN].client.locations
for location_id, location in client_locations.items():
for zone_id, zone in location.zones.items():
sensors.append(TotalConnectBinarySensor(zone_id, location_id, zone))
add_entities(sensors, True)
class TotalConnectBinarySensor(BinarySensorDevice):
"""Represent an TotalConnect zone."""
def __init__(self, zone_id, location_id, zone):
"""Initialize the TotalConnect status."""
self._zone_id = zone_id
self._location_id = location_id
self._zone = zone
self._name = self._zone.description
self._unique_id = f"{location_id} {zone_id}"
self._is_on = None
self._is_tampered = None
self._is_low_battery = None
@property
def unique_id(self):
"""Return the unique id."""
return self._unique_id
@property
def name(self):
"""Return the name of the device."""
return self._name
def update(self):
"""Return the state of the device."""
self._is_tampered = self._zone.is_tampered()
self._is_low_battery = self._zone.is_low_battery()
if self._zone.is_faulted() or self._zone.is_triggered():
self._is_on = True
else:
self._is_on = False
@property
def is_on(self):
"""Return true if the binary sensor is on."""
return self._is_on
@property
def device_class(self):
"""Return the class of this device, from component DEVICE_CLASSES."""
if self._zone.is_type_security():
return DEVICE_CLASS_DOOR
if self._zone.is_type_fire():
return DEVICE_CLASS_SMOKE
if self._zone.is_type_carbon_monoxide():
return DEVICE_CLASS_GAS
return None
@property
def device_state_attributes(self):
"""Return the state attributes."""
attributes = {
"zone_id": self._zone_id,
"location_id": self._location_id,
"low_battery": self._is_low_battery,
"tampered": self._is_tampered,
}
return attributes

View file

@ -2,7 +2,7 @@
"domain": "totalconnect", "domain": "totalconnect",
"name": "Honeywell Total Connect Alarm", "name": "Honeywell Total Connect Alarm",
"documentation": "https://www.home-assistant.io/integrations/totalconnect", "documentation": "https://www.home-assistant.io/integrations/totalconnect",
"requirements": ["total_connect_client==0.28"], "requirements": ["total_connect_client==0.50"],
"dependencies": [], "dependencies": [],
"codeowners": [] "codeowners": ["@austinmroczek"]
} }

View file

@ -1981,7 +1981,7 @@ todoist-python==8.0.0
toonapilib==3.2.4 toonapilib==3.2.4
# homeassistant.components.totalconnect # homeassistant.components.totalconnect
total_connect_client==0.28 total_connect_client==0.50
# homeassistant.components.tplink_lte # homeassistant.components.tplink_lte
tp-connected==0.0.4 tp-connected==0.0.4