2019-04-03 15:40:03 +00:00
|
|
|
"""Support for MyQ-Enabled Garage Doors."""
|
2021-02-10 20:30:52 +00:00
|
|
|
import logging
|
2019-10-21 20:46:39 +00:00
|
|
|
|
2020-05-10 18:42:28 +00:00
|
|
|
from pymyq.const import (
|
|
|
|
DEVICE_STATE as MYQ_DEVICE_STATE,
|
|
|
|
DEVICE_STATE_ONLINE as MYQ_DEVICE_STATE_ONLINE,
|
|
|
|
DEVICE_TYPE_GATE as MYQ_DEVICE_TYPE_GATE,
|
|
|
|
KNOWN_MODELS,
|
|
|
|
MANUFACTURER,
|
|
|
|
)
|
2021-02-10 20:30:52 +00:00
|
|
|
from pymyq.errors import MyQError
|
2017-02-13 10:20:07 +00:00
|
|
|
|
2018-08-29 12:33:09 +00:00
|
|
|
from homeassistant.components.cover import (
|
2020-03-22 23:28:55 +00:00
|
|
|
DEVICE_CLASS_GARAGE,
|
|
|
|
DEVICE_CLASS_GATE,
|
2019-07-31 19:25:30 +00:00
|
|
|
SUPPORT_CLOSE,
|
|
|
|
SUPPORT_OPEN,
|
2020-04-25 16:07:15 +00:00
|
|
|
CoverEntity,
|
2019-04-10 21:24:12 +00:00
|
|
|
)
|
2021-02-10 20:30:52 +00:00
|
|
|
from homeassistant.const import STATE_CLOSED, STATE_CLOSING, STATE_OPEN, STATE_OPENING
|
2020-08-30 12:40:00 +00:00
|
|
|
from homeassistant.helpers.update_coordinator import CoordinatorEntity
|
2020-03-22 23:28:55 +00:00
|
|
|
|
2021-02-10 20:30:52 +00:00
|
|
|
from .const import DOMAIN, MYQ_COORDINATOR, MYQ_GATEWAY, MYQ_TO_HASS
|
|
|
|
|
|
|
|
_LOGGER = logging.getLogger(__name__)
|
2017-02-13 10:20:07 +00:00
|
|
|
|
2020-03-20 20:28:14 +00:00
|
|
|
|
|
|
|
async def async_setup_entry(hass, config_entry, async_add_entities):
|
|
|
|
"""Set up mysq covers."""
|
2020-03-22 23:28:55 +00:00
|
|
|
data = hass.data[DOMAIN][config_entry.entry_id]
|
|
|
|
myq = data[MYQ_GATEWAY]
|
|
|
|
coordinator = data[MYQ_COORDINATOR]
|
|
|
|
|
|
|
|
async_add_entities(
|
|
|
|
[MyQDevice(coordinator, device) for device in myq.covers.values()], True
|
|
|
|
)
|
2017-02-13 10:20:07 +00:00
|
|
|
|
|
|
|
|
2020-08-30 12:40:00 +00:00
|
|
|
class MyQDevice(CoordinatorEntity, CoverEntity):
|
2017-02-13 10:20:07 +00:00
|
|
|
"""Representation of a MyQ cover."""
|
|
|
|
|
2020-03-22 23:28:55 +00:00
|
|
|
def __init__(self, coordinator, device):
|
2017-02-13 10:20:07 +00:00
|
|
|
"""Initialize with API object, device id."""
|
2020-08-30 12:40:00 +00:00
|
|
|
super().__init__(coordinator)
|
2018-11-18 17:37:03 +00:00
|
|
|
self._device = device
|
2017-02-13 10:20:07 +00:00
|
|
|
|
2018-05-24 05:58:35 +00:00
|
|
|
@property
|
|
|
|
def device_class(self):
|
|
|
|
"""Define this cover as a garage door."""
|
2021-02-10 20:30:52 +00:00
|
|
|
device_type = self._device.device_type
|
2020-03-22 23:28:55 +00:00
|
|
|
if device_type is not None and device_type == MYQ_DEVICE_TYPE_GATE:
|
|
|
|
return DEVICE_CLASS_GATE
|
|
|
|
return DEVICE_CLASS_GARAGE
|
2018-05-24 05:58:35 +00:00
|
|
|
|
2017-02-13 10:20:07 +00:00
|
|
|
@property
|
|
|
|
def name(self):
|
|
|
|
"""Return the name of the garage door if any."""
|
2018-11-18 17:37:03 +00:00
|
|
|
return self._device.name
|
2017-02-13 10:20:07 +00:00
|
|
|
|
2020-03-20 20:28:14 +00:00
|
|
|
@property
|
|
|
|
def available(self):
|
|
|
|
"""Return if the device is online."""
|
2020-08-30 12:40:00 +00:00
|
|
|
if not self.coordinator.last_update_success:
|
2020-03-22 23:28:55 +00:00
|
|
|
return False
|
|
|
|
|
2020-03-20 20:28:14 +00:00
|
|
|
# Not all devices report online so assume True if its missing
|
|
|
|
return self._device.device_json[MYQ_DEVICE_STATE].get(
|
|
|
|
MYQ_DEVICE_STATE_ONLINE, True
|
|
|
|
)
|
|
|
|
|
2017-02-13 10:20:07 +00:00
|
|
|
@property
|
|
|
|
def is_closed(self):
|
2017-04-30 05:04:49 +00:00
|
|
|
"""Return true if cover is closed, else False."""
|
2018-11-18 17:37:03 +00:00
|
|
|
return MYQ_TO_HASS.get(self._device.state) == STATE_CLOSED
|
2018-08-29 12:33:09 +00:00
|
|
|
|
|
|
|
@property
|
|
|
|
def is_closing(self):
|
|
|
|
"""Return if the cover is closing or not."""
|
2018-11-18 17:37:03 +00:00
|
|
|
return MYQ_TO_HASS.get(self._device.state) == STATE_CLOSING
|
2018-08-29 12:33:09 +00:00
|
|
|
|
2021-02-10 20:30:52 +00:00
|
|
|
@property
|
|
|
|
def is_open(self):
|
|
|
|
"""Return if the cover is opening or not."""
|
|
|
|
return MYQ_TO_HASS.get(self._device.state) == STATE_OPEN
|
|
|
|
|
2018-08-29 12:33:09 +00:00
|
|
|
@property
|
|
|
|
def is_opening(self):
|
|
|
|
"""Return if the cover is opening or not."""
|
2018-11-18 17:37:03 +00:00
|
|
|
return MYQ_TO_HASS.get(self._device.state) == STATE_OPENING
|
2017-02-13 10:20:07 +00:00
|
|
|
|
2018-08-29 12:33:09 +00:00
|
|
|
@property
|
|
|
|
def supported_features(self):
|
|
|
|
"""Flag supported features."""
|
|
|
|
return SUPPORT_OPEN | SUPPORT_CLOSE
|
|
|
|
|
2018-08-31 14:47:37 +00:00
|
|
|
@property
|
|
|
|
def unique_id(self):
|
2020-01-05 12:09:17 +00:00
|
|
|
"""Return a unique, Home Assistant friendly identifier for this entity."""
|
2018-11-18 17:37:03 +00:00
|
|
|
return self._device.device_id
|
|
|
|
|
|
|
|
async def async_close_cover(self, **kwargs):
|
|
|
|
"""Issue close command to cover."""
|
2021-02-10 20:30:52 +00:00
|
|
|
if self.is_closing or self.is_closed:
|
|
|
|
return
|
|
|
|
|
|
|
|
try:
|
|
|
|
wait_task = await self._device.close(wait_for_state=False)
|
|
|
|
except MyQError as err:
|
|
|
|
_LOGGER.error(
|
|
|
|
"Closing of cover %s failed with error: %s", self._device.name, str(err)
|
|
|
|
)
|
|
|
|
|
|
|
|
return
|
|
|
|
|
|
|
|
# Write closing state to HASS
|
|
|
|
self.async_write_ha_state()
|
|
|
|
|
|
|
|
if not await wait_task:
|
|
|
|
_LOGGER.error("Closing of cover %s failed", self._device.name)
|
|
|
|
|
|
|
|
# Write final state to HASS
|
|
|
|
self.async_write_ha_state()
|
2018-11-18 17:37:03 +00:00
|
|
|
|
|
|
|
async def async_open_cover(self, **kwargs):
|
|
|
|
"""Issue open command to cover."""
|
2021-02-10 20:30:52 +00:00
|
|
|
if self.is_opening or self.is_open:
|
|
|
|
return
|
2020-03-22 23:28:55 +00:00
|
|
|
|
2021-02-10 20:30:52 +00:00
|
|
|
try:
|
|
|
|
wait_task = await self._device.open(wait_for_state=False)
|
|
|
|
except MyQError as err:
|
|
|
|
_LOGGER.error(
|
|
|
|
"Opening of cover %s failed with error: %s", self._device.name, str(err)
|
|
|
|
)
|
|
|
|
return
|
|
|
|
|
|
|
|
# Write opening state to HASS
|
2020-03-20 20:28:14 +00:00
|
|
|
self.async_write_ha_state()
|
2018-08-31 14:47:37 +00:00
|
|
|
|
2021-02-10 20:30:52 +00:00
|
|
|
if not await wait_task:
|
|
|
|
_LOGGER.error("Opening of cover %s failed", self._device.name)
|
2020-03-22 23:28:55 +00:00
|
|
|
|
2021-02-10 20:30:52 +00:00
|
|
|
# Write final state to HASS
|
|
|
|
self.async_write_ha_state()
|
2020-03-20 20:28:14 +00:00
|
|
|
|
|
|
|
@property
|
|
|
|
def device_info(self):
|
|
|
|
"""Return the device_info of the device."""
|
|
|
|
device_info = {
|
|
|
|
"identifiers": {(DOMAIN, self._device.device_id)},
|
|
|
|
"name": self._device.name,
|
2020-03-31 18:58:44 +00:00
|
|
|
"manufacturer": MANUFACTURER,
|
2020-03-20 20:28:14 +00:00
|
|
|
"sw_version": self._device.firmware_version,
|
|
|
|
}
|
2020-03-31 18:58:44 +00:00
|
|
|
model = KNOWN_MODELS.get(self._device.device_id[2:4])
|
|
|
|
if model:
|
|
|
|
device_info["model"] = model
|
2020-03-20 20:28:14 +00:00
|
|
|
if self._device.parent_device_id:
|
|
|
|
device_info["via_device"] = (DOMAIN, self._device.parent_device_id)
|
|
|
|
return device_info
|
2020-03-22 23:28:55 +00:00
|
|
|
|
|
|
|
async def async_added_to_hass(self):
|
|
|
|
"""Subscribe to updates."""
|
2020-04-20 18:31:50 +00:00
|
|
|
self.async_on_remove(
|
2021-02-10 20:30:52 +00:00
|
|
|
self.coordinator.async_add_listener(self.async_write_ha_state)
|
2020-04-20 18:31:50 +00:00
|
|
|
)
|