2017-08-31 04:13:02 +00:00
|
|
|
"""
|
|
|
|
Support for Tesla cars.
|
|
|
|
|
|
|
|
For more details about this component, please refer to the documentation at
|
|
|
|
https://home-assistant.io/components/tesla/
|
|
|
|
"""
|
|
|
|
from collections import defaultdict
|
2017-09-07 16:20:27 +00:00
|
|
|
import logging
|
2018-01-21 06:35:38 +00:00
|
|
|
|
2017-08-31 04:13:02 +00:00
|
|
|
import voluptuous as vol
|
|
|
|
|
|
|
|
from homeassistant.const import (
|
2018-01-21 06:35:38 +00:00
|
|
|
ATTR_BATTERY_LEVEL, CONF_PASSWORD, CONF_SCAN_INTERVAL, CONF_USERNAME)
|
2017-08-31 04:13:02 +00:00
|
|
|
from homeassistant.helpers import config_validation as cv
|
2018-01-21 06:35:38 +00:00
|
|
|
from homeassistant.helpers import discovery
|
2017-08-31 04:13:02 +00:00
|
|
|
from homeassistant.helpers.entity import Entity
|
|
|
|
from homeassistant.util import slugify
|
|
|
|
|
2018-02-08 07:34:26 +00:00
|
|
|
REQUIREMENTS = ['teslajsonpy==0.0.23']
|
2017-08-31 04:13:02 +00:00
|
|
|
|
|
|
|
DOMAIN = 'tesla'
|
|
|
|
|
2017-09-07 16:20:27 +00:00
|
|
|
_LOGGER = logging.getLogger(__name__)
|
|
|
|
|
2017-08-31 04:13:02 +00:00
|
|
|
TESLA_ID_FORMAT = '{}_{}'
|
|
|
|
TESLA_ID_LIST_SCHEMA = vol.Schema([int])
|
|
|
|
|
|
|
|
CONFIG_SCHEMA = vol.Schema({
|
|
|
|
DOMAIN: vol.Schema({
|
|
|
|
vol.Required(CONF_USERNAME): cv.string,
|
|
|
|
vol.Required(CONF_PASSWORD): cv.string,
|
|
|
|
vol.Optional(CONF_SCAN_INTERVAL, default=300):
|
|
|
|
vol.All(cv.positive_int, vol.Clamp(min=300)),
|
|
|
|
}),
|
|
|
|
}, extra=vol.ALLOW_EXTRA)
|
|
|
|
|
2017-09-07 16:20:27 +00:00
|
|
|
NOTIFICATION_ID = 'tesla_integration_notification'
|
|
|
|
NOTIFICATION_TITLE = 'Tesla integration setup'
|
|
|
|
|
2017-08-31 04:13:02 +00:00
|
|
|
TESLA_COMPONENTS = [
|
2017-10-09 11:38:00 +00:00
|
|
|
'sensor', 'lock', 'climate', 'binary_sensor', 'device_tracker', 'switch'
|
2017-08-31 04:13:02 +00:00
|
|
|
]
|
|
|
|
|
|
|
|
|
|
|
|
def setup(hass, base_config):
|
2018-01-21 06:35:38 +00:00
|
|
|
"""Set up of Tesla component."""
|
2017-10-24 17:15:25 +00:00
|
|
|
from teslajsonpy import Controller as teslaAPI, TeslaException
|
2017-08-31 04:13:02 +00:00
|
|
|
|
|
|
|
config = base_config.get(DOMAIN)
|
|
|
|
|
|
|
|
email = config.get(CONF_USERNAME)
|
|
|
|
password = config.get(CONF_PASSWORD)
|
|
|
|
update_interval = config.get(CONF_SCAN_INTERVAL)
|
|
|
|
if hass.data.get(DOMAIN) is None:
|
2017-09-07 16:20:27 +00:00
|
|
|
try:
|
|
|
|
hass.data[DOMAIN] = {
|
2018-01-21 06:35:38 +00:00
|
|
|
'controller': teslaAPI(email, password, update_interval),
|
2017-09-07 16:20:27 +00:00
|
|
|
'devices': defaultdict(list)
|
|
|
|
}
|
|
|
|
_LOGGER.debug("Connected to the Tesla API.")
|
2017-10-24 17:15:25 +00:00
|
|
|
except TeslaException as ex:
|
2017-09-07 16:20:27 +00:00
|
|
|
if ex.code == 401:
|
|
|
|
hass.components.persistent_notification.create(
|
|
|
|
"Error:<br />Please check username and password."
|
|
|
|
"You will need to restart Home Assistant after fixing.",
|
|
|
|
title=NOTIFICATION_TITLE,
|
|
|
|
notification_id=NOTIFICATION_ID)
|
|
|
|
else:
|
|
|
|
hass.components.persistent_notification.create(
|
|
|
|
"Error:<br />Can't communicate with Tesla API.<br />"
|
|
|
|
"Error code: {} Reason: {}"
|
|
|
|
"You will need to restart Home Assistant after fixing."
|
2017-10-24 17:15:25 +00:00
|
|
|
"".format(ex.code, ex.message),
|
2017-09-07 16:20:27 +00:00
|
|
|
title=NOTIFICATION_TITLE,
|
|
|
|
notification_id=NOTIFICATION_ID)
|
|
|
|
_LOGGER.error("Unable to communicate with Tesla API: %s",
|
2017-10-24 17:15:25 +00:00
|
|
|
ex.message)
|
2017-09-07 16:20:27 +00:00
|
|
|
return False
|
2017-08-31 04:13:02 +00:00
|
|
|
|
|
|
|
all_devices = hass.data[DOMAIN]['controller'].list_vehicles()
|
|
|
|
|
|
|
|
if not all_devices:
|
|
|
|
return False
|
|
|
|
|
|
|
|
for device in all_devices:
|
|
|
|
hass.data[DOMAIN]['devices'][device.hass_type].append(device)
|
|
|
|
|
|
|
|
for component in TESLA_COMPONENTS:
|
|
|
|
discovery.load_platform(hass, component, DOMAIN, {}, base_config)
|
|
|
|
|
|
|
|
return True
|
|
|
|
|
|
|
|
|
|
|
|
class TeslaDevice(Entity):
|
|
|
|
"""Representation of a Tesla device."""
|
|
|
|
|
|
|
|
def __init__(self, tesla_device, controller):
|
2018-01-21 06:35:38 +00:00
|
|
|
"""Initialise of the Tesla device."""
|
2017-08-31 04:13:02 +00:00
|
|
|
self.tesla_device = tesla_device
|
|
|
|
self.controller = controller
|
|
|
|
self._name = self.tesla_device.name
|
|
|
|
self.tesla_id = slugify(self.tesla_device.uniq_name)
|
|
|
|
|
|
|
|
@property
|
|
|
|
def name(self):
|
|
|
|
"""Return the name of the device."""
|
|
|
|
return self._name
|
|
|
|
|
|
|
|
@property
|
|
|
|
def should_poll(self):
|
2018-01-21 06:35:38 +00:00
|
|
|
"""Return the polling state."""
|
2017-08-31 04:13:02 +00:00
|
|
|
return self.tesla_device.should_poll
|
|
|
|
|
|
|
|
@property
|
|
|
|
def device_state_attributes(self):
|
|
|
|
"""Return the state attributes of the device."""
|
|
|
|
attr = {}
|
|
|
|
|
|
|
|
if self.tesla_device.has_battery():
|
|
|
|
attr[ATTR_BATTERY_LEVEL] = self.tesla_device.battery_level()
|
|
|
|
return attr
|