195 lines
		
	
	
		
			5.8 KiB
		
	
	
	
		
			Python
		
	
	
			
		
		
	
	
			195 lines
		
	
	
		
			5.8 KiB
		
	
	
	
		
			Python
		
	
	
"""Platform for the opengarage.io cover component."""
 | 
						|
import logging
 | 
						|
 | 
						|
import opengarage
 | 
						|
import voluptuous as vol
 | 
						|
 | 
						|
from homeassistant.components.cover import (
 | 
						|
    DEVICE_CLASS_GARAGE,
 | 
						|
    PLATFORM_SCHEMA,
 | 
						|
    SUPPORT_CLOSE,
 | 
						|
    SUPPORT_OPEN,
 | 
						|
    CoverEntity,
 | 
						|
)
 | 
						|
from homeassistant.const import (
 | 
						|
    CONF_COVERS,
 | 
						|
    CONF_HOST,
 | 
						|
    CONF_NAME,
 | 
						|
    CONF_PORT,
 | 
						|
    CONF_SSL,
 | 
						|
    CONF_VERIFY_SSL,
 | 
						|
    STATE_CLOSED,
 | 
						|
    STATE_CLOSING,
 | 
						|
    STATE_OPEN,
 | 
						|
    STATE_OPENING,
 | 
						|
)
 | 
						|
from homeassistant.helpers.aiohttp_client import async_get_clientsession
 | 
						|
import homeassistant.helpers.config_validation as cv
 | 
						|
from homeassistant.helpers.device_registry import format_mac
 | 
						|
 | 
						|
_LOGGER = logging.getLogger(__name__)
 | 
						|
 | 
						|
ATTR_DISTANCE_SENSOR = "distance_sensor"
 | 
						|
ATTR_DOOR_STATE = "door_state"
 | 
						|
ATTR_SIGNAL_STRENGTH = "wifi_signal"
 | 
						|
 | 
						|
CONF_DEVICE_KEY = "device_key"
 | 
						|
 | 
						|
DEFAULT_NAME = "OpenGarage"
 | 
						|
DEFAULT_PORT = 80
 | 
						|
 | 
						|
STATES_MAP = {0: STATE_CLOSED, 1: STATE_OPEN}
 | 
						|
 | 
						|
COVER_SCHEMA = vol.Schema(
 | 
						|
    {
 | 
						|
        vol.Required(CONF_DEVICE_KEY): cv.string,
 | 
						|
        vol.Required(CONF_HOST): cv.string,
 | 
						|
        vol.Optional(CONF_NAME): cv.string,
 | 
						|
        vol.Optional(CONF_PORT, default=DEFAULT_PORT): cv.port,
 | 
						|
        vol.Optional(CONF_SSL, default=False): cv.boolean,
 | 
						|
        vol.Optional(CONF_VERIFY_SSL, default=True): cv.boolean,
 | 
						|
    }
 | 
						|
)
 | 
						|
 | 
						|
PLATFORM_SCHEMA = PLATFORM_SCHEMA.extend(
 | 
						|
    {vol.Required(CONF_COVERS): cv.schema_with_slug_keys(COVER_SCHEMA)}
 | 
						|
)
 | 
						|
 | 
						|
 | 
						|
async def async_setup_platform(hass, config, async_add_entities, discovery_info=None):
 | 
						|
    """Set up the OpenGarage covers."""
 | 
						|
    covers = []
 | 
						|
    devices = config.get(CONF_COVERS)
 | 
						|
 | 
						|
    for device_config in devices.values():
 | 
						|
        opengarage_url = (
 | 
						|
            f"{'https' if device_config[CONF_SSL] else 'http'}://"
 | 
						|
            f"{device_config.get(CONF_HOST)}:{device_config.get(CONF_PORT)}"
 | 
						|
        )
 | 
						|
 | 
						|
        open_garage = opengarage.OpenGarage(
 | 
						|
            opengarage_url,
 | 
						|
            device_config[CONF_DEVICE_KEY],
 | 
						|
            device_config[CONF_VERIFY_SSL],
 | 
						|
            async_get_clientsession(hass),
 | 
						|
        )
 | 
						|
        status = await open_garage.update_state()
 | 
						|
        covers.append(
 | 
						|
            OpenGarageCover(
 | 
						|
                device_config.get(CONF_NAME), open_garage, format_mac(status["mac"])
 | 
						|
            )
 | 
						|
        )
 | 
						|
 | 
						|
    async_add_entities(covers, True)
 | 
						|
 | 
						|
 | 
						|
class OpenGarageCover(CoverEntity):
 | 
						|
    """Representation of a OpenGarage cover."""
 | 
						|
 | 
						|
    def __init__(self, name, open_garage, device_id):
 | 
						|
        """Initialize the cover."""
 | 
						|
        self._name = name
 | 
						|
        self._open_garage = open_garage
 | 
						|
        self._state = None
 | 
						|
        self._state_before_move = None
 | 
						|
        self._extra_state_attributes = {}
 | 
						|
        self._available = True
 | 
						|
        self._device_id = device_id
 | 
						|
 | 
						|
    @property
 | 
						|
    def name(self):
 | 
						|
        """Return the name of the cover."""
 | 
						|
        return self._name
 | 
						|
 | 
						|
    @property
 | 
						|
    def available(self):
 | 
						|
        """Return True if entity is available."""
 | 
						|
        return self._available
 | 
						|
 | 
						|
    @property
 | 
						|
    def extra_state_attributes(self):
 | 
						|
        """Return the device state attributes."""
 | 
						|
        return self._extra_state_attributes
 | 
						|
 | 
						|
    @property
 | 
						|
    def is_closed(self):
 | 
						|
        """Return if the cover is closed."""
 | 
						|
        if self._state is None:
 | 
						|
            return None
 | 
						|
        return self._state in [STATE_CLOSED, STATE_OPENING]
 | 
						|
 | 
						|
    async def async_close_cover(self, **kwargs):
 | 
						|
        """Close the cover."""
 | 
						|
        if self._state in [STATE_CLOSED, STATE_CLOSING]:
 | 
						|
            return
 | 
						|
        self._state_before_move = self._state
 | 
						|
        self._state = STATE_CLOSING
 | 
						|
        await self._push_button()
 | 
						|
 | 
						|
    async def async_open_cover(self, **kwargs):
 | 
						|
        """Open the cover."""
 | 
						|
        if self._state in [STATE_OPEN, STATE_OPENING]:
 | 
						|
            return
 | 
						|
        self._state_before_move = self._state
 | 
						|
        self._state = STATE_OPENING
 | 
						|
        await self._push_button()
 | 
						|
 | 
						|
    async def async_update(self):
 | 
						|
        """Get updated status from API."""
 | 
						|
        status = await self._open_garage.update_state()
 | 
						|
        if status is None:
 | 
						|
            _LOGGER.error("Unable to connect to OpenGarage device")
 | 
						|
            self._available = False
 | 
						|
            return
 | 
						|
 | 
						|
        if self._name is None and status["name"] is not None:
 | 
						|
            self._name = status["name"]
 | 
						|
        state = STATES_MAP.get(status.get("door"))
 | 
						|
        if self._state_before_move is not None:
 | 
						|
            if self._state_before_move != state:
 | 
						|
                self._state = state
 | 
						|
                self._state_before_move = None
 | 
						|
        else:
 | 
						|
            self._state = state
 | 
						|
 | 
						|
        _LOGGER.debug("%s status: %s", self._name, self._state)
 | 
						|
        if status.get("rssi") is not None:
 | 
						|
            self._extra_state_attributes[ATTR_SIGNAL_STRENGTH] = status.get("rssi")
 | 
						|
        if status.get("dist") is not None:
 | 
						|
            self._extra_state_attributes[ATTR_DISTANCE_SENSOR] = status.get("dist")
 | 
						|
        if self._state is not None:
 | 
						|
            self._extra_state_attributes[ATTR_DOOR_STATE] = self._state
 | 
						|
 | 
						|
        self._available = True
 | 
						|
 | 
						|
    async def _push_button(self):
 | 
						|
        """Send commands to API."""
 | 
						|
        result = await self._open_garage.push_button()
 | 
						|
        if result is None:
 | 
						|
            _LOGGER.error("Unable to connect to OpenGarage device")
 | 
						|
        if result == 1:
 | 
						|
            return
 | 
						|
 | 
						|
        if result == 2:
 | 
						|
            _LOGGER.error("Unable to control %s: Device key is incorrect", self._name)
 | 
						|
        elif result > 2:
 | 
						|
            _LOGGER.error("Unable to control %s: Error code %s", self._name, result)
 | 
						|
 | 
						|
        self._state = self._state_before_move
 | 
						|
        self._state_before_move = None
 | 
						|
 | 
						|
    @property
 | 
						|
    def device_class(self):
 | 
						|
        """Return the class of this device, from component DEVICE_CLASSES."""
 | 
						|
        return DEVICE_CLASS_GARAGE
 | 
						|
 | 
						|
    @property
 | 
						|
    def supported_features(self):
 | 
						|
        """Flag supported features."""
 | 
						|
        return SUPPORT_OPEN | SUPPORT_CLOSE
 | 
						|
 | 
						|
    @property
 | 
						|
    def unique_id(self):
 | 
						|
        """Return a unique ID."""
 | 
						|
        return self._device_id
 |