165 lines
4.9 KiB
Python
165 lines
4.9 KiB
Python
"""Support for Soma Smartshades."""
|
|
import logging
|
|
|
|
from api.soma_api import SomaApi
|
|
from requests import RequestException
|
|
import voluptuous as vol
|
|
|
|
from homeassistant import config_entries
|
|
from homeassistant.config_entries import ConfigEntry
|
|
from homeassistant.const import CONF_HOST, CONF_PORT, Platform
|
|
from homeassistant.core import HomeAssistant
|
|
import homeassistant.helpers.config_validation as cv
|
|
from homeassistant.helpers.entity import DeviceInfo, Entity
|
|
from homeassistant.helpers.typing import ConfigType
|
|
|
|
from .const import API, DOMAIN, HOST, PORT
|
|
from .utils import is_api_response_success
|
|
|
|
_LOGGER = logging.getLogger(__name__)
|
|
|
|
DEVICES = "devices"
|
|
|
|
CONFIG_SCHEMA = vol.Schema(
|
|
vol.All(
|
|
cv.deprecated(DOMAIN),
|
|
{
|
|
DOMAIN: vol.Schema(
|
|
{vol.Required(CONF_HOST): cv.string, vol.Required(CONF_PORT): cv.string}
|
|
)
|
|
},
|
|
),
|
|
extra=vol.ALLOW_EXTRA,
|
|
)
|
|
|
|
PLATFORMS = [Platform.COVER, Platform.SENSOR]
|
|
|
|
|
|
async def async_setup(hass: HomeAssistant, config: ConfigType) -> bool:
|
|
"""Set up the Soma component."""
|
|
if DOMAIN not in config:
|
|
return True
|
|
|
|
hass.async_create_task(
|
|
hass.config_entries.flow.async_init(
|
|
DOMAIN,
|
|
data=config[DOMAIN],
|
|
context={"source": config_entries.SOURCE_IMPORT},
|
|
)
|
|
)
|
|
|
|
return True
|
|
|
|
|
|
async def async_setup_entry(hass: HomeAssistant, entry: ConfigEntry) -> bool:
|
|
"""Set up Soma from a config entry."""
|
|
hass.data[DOMAIN] = {}
|
|
api = await hass.async_add_executor_job(SomaApi, entry.data[HOST], entry.data[PORT])
|
|
devices = await hass.async_add_executor_job(api.list_devices)
|
|
hass.data[DOMAIN] = {API: api, DEVICES: devices["shades"]}
|
|
|
|
await hass.config_entries.async_forward_entry_setups(entry, PLATFORMS)
|
|
|
|
return True
|
|
|
|
|
|
async def async_unload_entry(hass: HomeAssistant, entry: ConfigEntry) -> bool:
|
|
"""Unload a config entry."""
|
|
return await hass.config_entries.async_unload_platforms(entry, PLATFORMS)
|
|
|
|
|
|
def soma_api_call(api_call):
|
|
"""Soma api call decorator."""
|
|
|
|
async def inner(self) -> dict:
|
|
response = {}
|
|
try:
|
|
response_from_api = await api_call(self)
|
|
except RequestException:
|
|
if self.api_is_available:
|
|
_LOGGER.warning("Connection to SOMA Connect failed")
|
|
self.api_is_available = False
|
|
else:
|
|
if not self.api_is_available:
|
|
self.api_is_available = True
|
|
_LOGGER.info("Connection to SOMA Connect succeeded")
|
|
|
|
if not is_api_response_success(response_from_api):
|
|
if self.is_available:
|
|
self.is_available = False
|
|
_LOGGER.warning(
|
|
(
|
|
"Device is unreachable (%s). Error while fetching the"
|
|
" state: %s"
|
|
),
|
|
self.name,
|
|
response_from_api["msg"],
|
|
)
|
|
else:
|
|
if not self.is_available:
|
|
self.is_available = True
|
|
_LOGGER.info("Device %s is now reachable", self.name)
|
|
response = response_from_api
|
|
return response
|
|
|
|
return inner
|
|
|
|
|
|
class SomaEntity(Entity):
|
|
"""Representation of a generic Soma device."""
|
|
|
|
def __init__(self, device, api):
|
|
"""Initialize the Soma device."""
|
|
self.device = device
|
|
self.api = api
|
|
self.current_position = 50
|
|
self.battery_state = 0
|
|
self.is_available = True
|
|
self.api_is_available = True
|
|
|
|
@property
|
|
def available(self):
|
|
"""Return true if the last API commands returned successfully."""
|
|
return self.is_available
|
|
|
|
@property
|
|
def unique_id(self):
|
|
"""Return the unique id base on the id returned by pysoma API."""
|
|
return self.device["mac"]
|
|
|
|
@property
|
|
def name(self):
|
|
"""Return the name of the device."""
|
|
return self.device["name"]
|
|
|
|
@property
|
|
def device_info(self) -> DeviceInfo:
|
|
"""Return device specific attributes.
|
|
|
|
Implemented by platform classes.
|
|
"""
|
|
return DeviceInfo(
|
|
identifiers={(DOMAIN, self.unique_id)},
|
|
manufacturer="Wazombi Labs",
|
|
name=self.name,
|
|
)
|
|
|
|
def set_position(self, position: int) -> None:
|
|
"""Set the current device position."""
|
|
self.current_position = position
|
|
self.schedule_update_ha_state()
|
|
|
|
@soma_api_call
|
|
async def get_shade_state_from_api(self) -> dict:
|
|
"""Return the shade state from the api."""
|
|
return await self.hass.async_add_executor_job(
|
|
self.api.get_shade_state, self.device["mac"]
|
|
)
|
|
|
|
@soma_api_call
|
|
async def get_battery_level_from_api(self) -> dict:
|
|
"""Return the battery level from the api."""
|
|
return await self.hass.async_add_executor_job(
|
|
self.api.get_battery_level, self.device["mac"]
|
|
)
|