"""The Tesla Powerwall integration.""" import asyncio from datetime import timedelta import logging import requests from tesla_powerwall import ApiError, Powerwall, PowerwallUnreachableError import voluptuous as vol from homeassistant.config_entries import SOURCE_IMPORT, ConfigEntry from homeassistant.const import CONF_IP_ADDRESS from homeassistant.core import HomeAssistant from homeassistant.exceptions import ConfigEntryNotReady import homeassistant.helpers.config_validation as cv from homeassistant.helpers.update_coordinator import DataUpdateCoordinator from .const import ( DOMAIN, POWERWALL_API_CHARGE, POWERWALL_API_DEVICE_TYPE, POWERWALL_API_GRID_STATUS, POWERWALL_API_METERS, POWERWALL_API_SITE_INFO, POWERWALL_API_SITEMASTER, POWERWALL_API_STATUS, POWERWALL_COORDINATOR, POWERWALL_HTTP_SESSION, POWERWALL_OBJECT, UPDATE_INTERVAL, ) CONFIG_SCHEMA = vol.Schema( {DOMAIN: vol.Schema({vol.Required(CONF_IP_ADDRESS): cv.string})}, extra=vol.ALLOW_EXTRA, ) PLATFORMS = ["binary_sensor", "sensor"] _LOGGER = logging.getLogger(__name__) async def async_setup(hass: HomeAssistant, config: dict): """Set up the Tesla Powerwall component.""" hass.data.setdefault(DOMAIN, {}) conf = config.get(DOMAIN) if not conf: return True hass.async_create_task( hass.config_entries.flow.async_init( DOMAIN, context={"source": SOURCE_IMPORT}, data=conf, ) ) return True async def async_setup_entry(hass: HomeAssistant, entry: ConfigEntry): """Set up Tesla Powerwall from a config entry.""" entry_id = entry.entry_id hass.data[DOMAIN].setdefault(entry_id, {}) http_session = requests.Session() power_wall = Powerwall(entry.data[CONF_IP_ADDRESS], http_session=http_session) try: powerwall_data = await hass.async_add_executor_job(call_base_info, power_wall) except (PowerwallUnreachableError, ApiError, ConnectionError): http_session.close() raise ConfigEntryNotReady async def async_update_data(): """Fetch data from API endpoint.""" return await hass.async_add_executor_job(_fetch_powerwall_data, power_wall) coordinator = DataUpdateCoordinator( hass, _LOGGER, name="Powerwall site", update_method=async_update_data, update_interval=timedelta(seconds=UPDATE_INTERVAL), ) hass.data[DOMAIN][entry.entry_id] = powerwall_data hass.data[DOMAIN][entry.entry_id].update( { POWERWALL_OBJECT: power_wall, POWERWALL_COORDINATOR: coordinator, POWERWALL_HTTP_SESSION: http_session, } ) await coordinator.async_refresh() for component in PLATFORMS: hass.async_create_task( hass.config_entries.async_forward_entry_setup(entry, component) ) return True def call_base_info(power_wall): """Wrap powerwall properties to be a callable.""" return { POWERWALL_API_SITE_INFO: power_wall.get_site_info(), POWERWALL_API_STATUS: power_wall.get_status(), POWERWALL_API_DEVICE_TYPE: power_wall.get_device_type(), } def _fetch_powerwall_data(power_wall): """Process and update powerwall data.""" return { POWERWALL_API_CHARGE: power_wall.get_charge(), POWERWALL_API_SITEMASTER: power_wall.get_sitemaster(), POWERWALL_API_METERS: power_wall.get_meters(), POWERWALL_API_GRID_STATUS: power_wall.get_grid_status(), } async def async_unload_entry(hass: HomeAssistant, entry: ConfigEntry): """Unload a config entry.""" unload_ok = all( await asyncio.gather( *[ hass.config_entries.async_forward_entry_unload(entry, component) for component in PLATFORMS ] ) ) hass.data[DOMAIN][entry.entry_id][POWERWALL_HTTP_SESSION].close() if unload_ok: hass.data[DOMAIN].pop(entry.entry_id) return unload_ok