180 lines
5.1 KiB
Python
180 lines
5.1 KiB
Python
"""Support for Sure Petcare cat/pet flaps."""
|
|
import logging
|
|
from typing import Any, Dict, List
|
|
|
|
from surepy import (
|
|
SurePetcare,
|
|
SurePetcareAuthenticationError,
|
|
SurePetcareError,
|
|
SureProductID,
|
|
)
|
|
import voluptuous as vol
|
|
|
|
from homeassistant.const import (
|
|
CONF_ID,
|
|
CONF_PASSWORD,
|
|
CONF_SCAN_INTERVAL,
|
|
CONF_TYPE,
|
|
CONF_USERNAME,
|
|
)
|
|
from homeassistant.helpers import config_validation as cv
|
|
from homeassistant.helpers.aiohttp_client import async_get_clientsession
|
|
from homeassistant.helpers.dispatcher import async_dispatcher_send
|
|
from homeassistant.helpers.event import async_track_time_interval
|
|
|
|
from .const import (
|
|
CONF_FEEDERS,
|
|
CONF_FLAPS,
|
|
CONF_PARENT,
|
|
CONF_PETS,
|
|
CONF_PRODUCT_ID,
|
|
DATA_SURE_PETCARE,
|
|
DEFAULT_SCAN_INTERVAL,
|
|
DOMAIN,
|
|
SPC,
|
|
TOPIC_UPDATE,
|
|
)
|
|
|
|
_LOGGER = logging.getLogger(__name__)
|
|
|
|
|
|
CONFIG_SCHEMA = vol.Schema(
|
|
{
|
|
DOMAIN: vol.Schema(
|
|
{
|
|
vol.Required(CONF_USERNAME): cv.string,
|
|
vol.Required(CONF_PASSWORD): cv.string,
|
|
vol.Optional(CONF_FEEDERS, default=[]): vol.All(
|
|
cv.ensure_list, [cv.positive_int]
|
|
),
|
|
vol.Optional(CONF_FLAPS, default=[]): vol.All(
|
|
cv.ensure_list, [cv.positive_int]
|
|
),
|
|
vol.Optional(CONF_PETS): vol.All(cv.ensure_list, [cv.positive_int]),
|
|
vol.Optional(
|
|
CONF_SCAN_INTERVAL, default=DEFAULT_SCAN_INTERVAL
|
|
): cv.time_period,
|
|
}
|
|
)
|
|
},
|
|
extra=vol.ALLOW_EXTRA,
|
|
)
|
|
|
|
|
|
async def async_setup(hass, config) -> bool:
|
|
"""Initialize the Sure Petcare component."""
|
|
conf = config[DOMAIN]
|
|
|
|
# update interval
|
|
scan_interval = conf[CONF_SCAN_INTERVAL]
|
|
|
|
# shared data
|
|
hass.data[DOMAIN] = hass.data[DATA_SURE_PETCARE] = {}
|
|
|
|
# sure petcare api connection
|
|
try:
|
|
surepy = SurePetcare(
|
|
conf[CONF_USERNAME],
|
|
conf[CONF_PASSWORD],
|
|
hass.loop,
|
|
async_get_clientsession(hass),
|
|
)
|
|
await surepy.get_data()
|
|
except SurePetcareAuthenticationError:
|
|
_LOGGER.error("Unable to connect to surepetcare.io: Wrong credentials!")
|
|
return False
|
|
except SurePetcareError as error:
|
|
_LOGGER.error("Unable to connect to surepetcare.io: Wrong %s!", error)
|
|
return False
|
|
|
|
# add feeders
|
|
things = [
|
|
{CONF_ID: feeder, CONF_TYPE: SureProductID.FEEDER}
|
|
for feeder in conf[CONF_FEEDERS]
|
|
]
|
|
|
|
# add flaps (don't differentiate between CAT and PET for now)
|
|
things.extend(
|
|
[
|
|
{CONF_ID: flap, CONF_TYPE: SureProductID.PET_FLAP}
|
|
for flap in conf[CONF_FLAPS]
|
|
]
|
|
)
|
|
|
|
# discover hubs the flaps/feeders are connected to
|
|
for device in things.copy():
|
|
device_data = await surepy.device(device[CONF_ID])
|
|
if (
|
|
CONF_PARENT in device_data
|
|
and device_data[CONF_PARENT][CONF_PRODUCT_ID] == SureProductID.HUB
|
|
and device_data[CONF_PARENT][CONF_ID] not in things
|
|
):
|
|
things.append(
|
|
{
|
|
CONF_ID: device_data[CONF_PARENT][CONF_ID],
|
|
CONF_TYPE: SureProductID.HUB,
|
|
}
|
|
)
|
|
|
|
# add pets
|
|
things.extend(
|
|
[{CONF_ID: pet, CONF_TYPE: SureProductID.PET} for pet in conf[CONF_PETS]]
|
|
)
|
|
|
|
_LOGGER.debug("Devices and Pets to setup: %s", things)
|
|
|
|
spc = hass.data[DATA_SURE_PETCARE][SPC] = SurePetcareAPI(hass, surepy, things)
|
|
|
|
# initial update
|
|
await spc.async_update()
|
|
|
|
async_track_time_interval(hass, spc.async_update, scan_interval)
|
|
|
|
# load platforms
|
|
hass.async_create_task(
|
|
hass.helpers.discovery.async_load_platform("binary_sensor", DOMAIN, {}, config)
|
|
)
|
|
hass.async_create_task(
|
|
hass.helpers.discovery.async_load_platform("sensor", DOMAIN, {}, config)
|
|
)
|
|
|
|
return True
|
|
|
|
|
|
class SurePetcareAPI:
|
|
"""Define a generic Sure Petcare object."""
|
|
|
|
def __init__(self, hass, surepy: SurePetcare, ids: List[Dict[str, Any]]) -> None:
|
|
"""Initialize the Sure Petcare object."""
|
|
self.hass = hass
|
|
self.surepy = surepy
|
|
self.ids = ids
|
|
self.states: Dict[str, Any] = {}
|
|
|
|
async def async_update(self, arg: Any = None) -> None:
|
|
"""Refresh Sure Petcare data."""
|
|
|
|
await self.surepy.get_data()
|
|
|
|
for thing in self.ids:
|
|
sure_id = thing[CONF_ID]
|
|
sure_type = thing[CONF_TYPE]
|
|
|
|
try:
|
|
type_state = self.states.setdefault(sure_type, {})
|
|
|
|
if sure_type in [
|
|
SureProductID.CAT_FLAP,
|
|
SureProductID.PET_FLAP,
|
|
SureProductID.FEEDER,
|
|
SureProductID.HUB,
|
|
]:
|
|
type_state[sure_id] = await self.surepy.device(sure_id)
|
|
elif sure_type == SureProductID.PET:
|
|
type_state[sure_id] = await self.surepy.pet(sure_id)
|
|
|
|
except SurePetcareError as error:
|
|
_LOGGER.error("Unable to retrieve data from surepetcare.io: %s", error)
|
|
|
|
async_dispatcher_send(self.hass, TOPIC_UPDATE)
|