"""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)