""" Support for Tibber. For more details about this platform, please refer to the documentation at https://home-assistant.io/components/sensor.tibber/ """ import asyncio import logging from datetime import timedelta import aiohttp import voluptuous as vol import homeassistant.helpers.config_validation as cv from homeassistant.components.sensor import PLATFORM_SCHEMA from homeassistant.const import CONF_ACCESS_TOKEN from homeassistant.exceptions import PlatformNotReady from homeassistant.helpers.aiohttp_client import async_get_clientsession from homeassistant.helpers.entity import Entity from homeassistant.util import dt as dt_util REQUIREMENTS = ['pyTibber==0.4.0'] _LOGGER = logging.getLogger(__name__) PLATFORM_SCHEMA = PLATFORM_SCHEMA.extend({ vol.Required(CONF_ACCESS_TOKEN): cv.string }) ICON = 'mdi:currency-usd' SCAN_INTERVAL = timedelta(minutes=1) async def async_setup_platform(hass, config, async_add_devices, discovery_info=None): """Set up the Tibber sensor.""" import tibber tibber_connection = tibber.Tibber(config[CONF_ACCESS_TOKEN], websession=async_get_clientsession(hass)) try: await tibber_connection.update_info() dev = [] for home in tibber_connection.get_homes(): await home.update_info() dev.append(TibberSensor(home)) except (asyncio.TimeoutError, aiohttp.ClientError): raise PlatformNotReady() async_add_devices(dev, True) class TibberSensor(Entity): """Representation of an Tibber sensor.""" def __init__(self, tibber_home): """Initialize the sensor.""" self._tibber_home = tibber_home self._last_updated = None self._state = None self._device_state_attributes = {} self._unit_of_measurement = self._tibber_home.price_unit self._name = 'Electricity price {}'.format(tibber_home.address1) async def async_update(self): """Get the latest data and updates the states.""" now = dt_util.utcnow() if self._tibber_home.current_price_total and self._last_updated and \ dt_util.as_utc(dt_util.parse_datetime(self._last_updated)).hour\ == now.hour: return def _find_current_price(): for key, price_total in self._tibber_home.price_total.items(): price_time = dt_util.as_utc(dt_util.parse_datetime(key)) time_diff = (now - price_time).total_seconds()/60 if time_diff >= 0 and time_diff < 60: self._state = round(price_total, 3) self._last_updated = key return True return False if _find_current_price(): return _LOGGER.debug("No cached data found, so asking for new data") await self._tibber_home.update_info() await self._tibber_home.update_price_info() data = self._tibber_home.info['viewer']['home'] self._device_state_attributes['app_nickname'] = data['appNickname'] self._device_state_attributes['grid_company'] =\ data['meteringPointData']['gridCompany'] self._device_state_attributes['estimated_annual_consumption'] =\ data['meteringPointData']['estimatedAnnualConsumption'] _find_current_price() @property def device_state_attributes(self): """Return the state attributes.""" return self._device_state_attributes @property def name(self): """Return the name of the sensor.""" return self._name @property def state(self): """Return the state of the device.""" return self._state @property def icon(self): """Return the icon to use in the frontend.""" return ICON @property def unit_of_measurement(self): """Return the unit of measurement of this entity.""" return self._unit_of_measurement @property def unique_id(self): """Return a unique ID.""" home = self._tibber_home.info['viewer']['home'] return home['meteringPointData']['consumptionEan']