core/homeassistant/components/volvooncall.py

171 lines
5.5 KiB
Python

"""
Support for Volvo On Call.
For more details about this component, please refer to the documentation at
https://home-assistant.io/components/volvooncall/
"""
from datetime import timedelta
import logging
from homeassistant.const import (CONF_USERNAME, CONF_PASSWORD,
CONF_NAME, CONF_RESOURCES)
from homeassistant.helpers import discovery
import homeassistant.helpers.config_validation as cv
from homeassistant.helpers.entity import Entity
from homeassistant.helpers.event import track_point_in_utc_time
from homeassistant.util.dt import utcnow
import voluptuous as vol
DOMAIN = 'volvooncall'
REQUIREMENTS = ['volvooncall==0.3.3']
_LOGGER = logging.getLogger(__name__)
CONF_UPDATE_INTERVAL = 'update_interval'
MIN_UPDATE_INTERVAL = timedelta(minutes=1)
DEFAULT_UPDATE_INTERVAL = timedelta(minutes=1)
RESOURCES = {'position': ('device_tracker',),
'lock': ('lock', 'Lock'),
'heater': ('switch', 'Heater', 'mdi:radiator'),
'odometer': ('sensor', 'Odometer', 'mdi:speedometer', 'km'),
'fuel_amount': ('sensor', 'Fuel', 'mdi:gas-station', 'L'),
'fuel_amount_level': ('sensor', 'Fuel', 'mdi:water-percent', '%'),
'distance_to_empty': ('sensor', 'Range', 'mdi:ruler', 'km'),
'washer_fluid_level': ('binary_sensor', 'Washer fluid'),
'brake_fluid': ('binary_sensor', 'Brake Fluid'),
'service_warning_status': ('binary_sensor', 'Service'),
'bulb_failures': ('binary_sensor', 'Bulbs'),
'doors': ('binary_sensor', 'Doors'),
'windows': ('binary_sensor', 'Windows')}
CONFIG_SCHEMA = vol.Schema({
DOMAIN: vol.Schema({
vol.Required(CONF_USERNAME): cv.string,
vol.Required(CONF_PASSWORD): cv.string,
vol.Optional(CONF_UPDATE_INTERVAL, default=DEFAULT_UPDATE_INTERVAL): (
vol.All(cv.time_period, vol.Clamp(min=MIN_UPDATE_INTERVAL))),
vol.Optional(CONF_NAME, default={}): vol.Schema(
{cv.slug: cv.string}),
vol.Optional(CONF_RESOURCES): vol.All(
cv.ensure_list, [vol.In(RESOURCES)]),
}),
}, extra=vol.ALLOW_EXTRA)
def setup(hass, config):
"""Setup the VOC component."""
from volvooncall import Connection
connection = Connection(
config[DOMAIN].get(CONF_USERNAME),
config[DOMAIN].get(CONF_PASSWORD))
interval = config[DOMAIN].get(CONF_UPDATE_INTERVAL)
class state: # pylint:disable=invalid-name
"""Namespace to hold state for each vehicle."""
entities = {}
vehicles = {}
names = config[DOMAIN].get(CONF_NAME)
hass.data[DOMAIN] = state
def discover_vehicle(vehicle):
"""Load relevant platforms."""
state.entities[vehicle.vin] = []
for attr, (component, *_) in RESOURCES.items():
if (getattr(vehicle, attr + '_supported', True) and
attr in config[DOMAIN].get(CONF_RESOURCES, [attr])):
discovery.load_platform(hass,
component,
DOMAIN,
(vehicle.vin, attr),
config)
def update_vehicle(vehicle):
"""Updated information on vehicle received."""
state.vehicles[vehicle.vin] = vehicle
if vehicle.vin not in state.entities:
discover_vehicle(vehicle)
for entity in state.entities[vehicle.vin]:
if isinstance(entity, Entity):
entity.schedule_update_ha_state()
else:
entity(vehicle) # device tracker
def update(now):
"""Update status from the online service."""
try:
if not connection.update():
_LOGGER.warning('Could not query server')
return False
for vehicle in connection.vehicles:
update_vehicle(vehicle)
return True
finally:
track_point_in_utc_time(hass, update, utcnow() + interval)
_LOGGER.info('Logging in to service')
return update(utcnow())
class VolvoEntity(Entity):
"""Base class for all VOC entities."""
def __init__(self, hass, vin, attribute):
"""Initialize the entity."""
self._hass = hass
self._vin = vin
self._attribute = attribute
self._state.entities[self._vin].append(self)
@property
def _state(self):
return self._hass.data[DOMAIN]
@property
def vehicle(self):
"""Return vehicle."""
return self._state.vehicles[self._vin]
@property
def _vehicle_name(self):
return (self._state.names.get(self._vin.lower()) or
self._state.names.get(
self.vehicle.registration_number.lower()) or
self.vehicle.registration_number)
@property
def _entity_name(self):
return RESOURCES[self._attribute][1]
@property
def name(self):
"""Return full name of the entity."""
return '%s %s' % (
self._vehicle_name,
self._entity_name)
@property
def should_poll(self):
"""Polling is not needed."""
return False
@property
def assumed_state(self):
"""Return true if unable to access real state of entity."""
return True
@property
def device_state_attributes(self):
"""Return device specific state attributes."""
return dict(model='%s/%s' % (
self.vehicle.vehicle_type,
self.vehicle.model_year))