2019-04-03 15:40:03 +00:00
|
|
|
"""Interfaces with the myLeviton API for Decora Smart WiFi products."""
|
2017-08-06 18:30:28 +00:00
|
|
|
|
|
|
|
import logging
|
|
|
|
|
|
|
|
import voluptuous as vol
|
|
|
|
|
|
|
|
from homeassistant.components.light import (
|
2019-07-31 19:25:30 +00:00
|
|
|
ATTR_BRIGHTNESS,
|
|
|
|
ATTR_TRANSITION,
|
|
|
|
Light,
|
|
|
|
PLATFORM_SCHEMA,
|
|
|
|
SUPPORT_BRIGHTNESS,
|
|
|
|
SUPPORT_TRANSITION,
|
|
|
|
)
|
|
|
|
from homeassistant.const import CONF_USERNAME, CONF_PASSWORD, EVENT_HOMEASSISTANT_STOP
|
2017-08-06 18:30:28 +00:00
|
|
|
import homeassistant.helpers.config_validation as cv
|
|
|
|
|
|
|
|
_LOGGER = logging.getLogger(__name__)
|
|
|
|
|
|
|
|
# Validation of the user's configuration
|
2019-07-31 19:25:30 +00:00
|
|
|
PLATFORM_SCHEMA = PLATFORM_SCHEMA.extend(
|
|
|
|
{vol.Required(CONF_USERNAME): cv.string, vol.Required(CONF_PASSWORD): cv.string}
|
|
|
|
)
|
2017-08-06 18:30:28 +00:00
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
NOTIFICATION_ID = "leviton_notification"
|
|
|
|
NOTIFICATION_TITLE = "myLeviton Decora Setup"
|
2017-08-06 18:30:28 +00:00
|
|
|
|
|
|
|
|
2018-08-24 14:37:30 +00:00
|
|
|
def setup_platform(hass, config, add_entities, discovery_info=None):
|
2017-08-06 18:30:28 +00:00
|
|
|
"""Set up the Decora WiFi platform."""
|
2018-06-25 17:05:07 +00:00
|
|
|
# pylint: disable=import-error, no-name-in-module
|
2017-08-06 18:30:28 +00:00
|
|
|
from decora_wifi import DecoraWiFiSession
|
|
|
|
from decora_wifi.models.person import Person
|
|
|
|
from decora_wifi.models.residential_account import ResidentialAccount
|
2018-11-27 15:53:28 +00:00
|
|
|
from decora_wifi.models.residence import Residence
|
2017-08-06 18:30:28 +00:00
|
|
|
|
|
|
|
email = config.get(CONF_USERNAME)
|
|
|
|
password = config.get(CONF_PASSWORD)
|
|
|
|
session = DecoraWiFiSession()
|
|
|
|
|
|
|
|
try:
|
|
|
|
success = session.login(email, password)
|
|
|
|
|
|
|
|
# If login failed, notify user.
|
|
|
|
if success is None:
|
2019-07-31 19:25:30 +00:00
|
|
|
msg = "Failed to log into myLeviton Services. Check credentials."
|
2017-08-06 18:30:28 +00:00
|
|
|
_LOGGER.error(msg)
|
|
|
|
hass.components.persistent_notification.create(
|
2019-07-31 19:25:30 +00:00
|
|
|
msg, title=NOTIFICATION_TITLE, notification_id=NOTIFICATION_ID
|
|
|
|
)
|
2017-08-06 18:30:28 +00:00
|
|
|
return False
|
|
|
|
|
|
|
|
# Gather all the available devices...
|
|
|
|
perms = session.user.get_residential_permissions()
|
|
|
|
all_switches = []
|
|
|
|
for permission in perms:
|
2018-11-27 15:53:28 +00:00
|
|
|
if permission.residentialAccountId is not None:
|
2019-07-31 19:25:30 +00:00
|
|
|
acct = ResidentialAccount(session, permission.residentialAccountId)
|
2018-11-27 15:53:28 +00:00
|
|
|
for residence in acct.get_residences():
|
|
|
|
for switch in residence.get_iot_switches():
|
|
|
|
all_switches.append(switch)
|
|
|
|
elif permission.residenceId is not None:
|
|
|
|
residence = Residence(session, permission.residenceId)
|
2017-08-06 18:30:28 +00:00
|
|
|
for switch in residence.get_iot_switches():
|
|
|
|
all_switches.append(switch)
|
|
|
|
|
2018-08-24 14:37:30 +00:00
|
|
|
add_entities(DecoraWifiLight(sw) for sw in all_switches)
|
2017-08-06 18:30:28 +00:00
|
|
|
except ValueError:
|
2019-07-31 19:25:30 +00:00
|
|
|
_LOGGER.error("Failed to communicate with myLeviton Service.")
|
2017-08-06 18:30:28 +00:00
|
|
|
|
|
|
|
# Listen for the stop event and log out.
|
|
|
|
def logout(event):
|
|
|
|
"""Log out..."""
|
|
|
|
try:
|
|
|
|
if session is not None:
|
|
|
|
Person.logout(session)
|
|
|
|
except ValueError:
|
2019-07-31 19:25:30 +00:00
|
|
|
_LOGGER.error("Failed to log out of myLeviton Service.")
|
2017-08-06 18:30:28 +00:00
|
|
|
|
|
|
|
hass.bus.listen(EVENT_HOMEASSISTANT_STOP, logout)
|
|
|
|
|
|
|
|
|
|
|
|
class DecoraWifiLight(Light):
|
|
|
|
"""Representation of a Decora WiFi switch."""
|
|
|
|
|
|
|
|
def __init__(self, switch):
|
|
|
|
"""Initialize the switch."""
|
|
|
|
self._switch = switch
|
|
|
|
|
|
|
|
@property
|
|
|
|
def supported_features(self):
|
|
|
|
"""Return supported features."""
|
|
|
|
if self._switch.canSetLevel:
|
|
|
|
return SUPPORT_BRIGHTNESS | SUPPORT_TRANSITION
|
2018-02-11 17:20:28 +00:00
|
|
|
return 0
|
2017-08-06 18:30:28 +00:00
|
|
|
|
|
|
|
@property
|
|
|
|
def name(self):
|
|
|
|
"""Return the display name of this switch."""
|
|
|
|
return self._switch.name
|
|
|
|
|
|
|
|
@property
|
|
|
|
def brightness(self):
|
|
|
|
"""Return the brightness of the dimmer switch."""
|
|
|
|
return int(self._switch.brightness * 255 / 100)
|
|
|
|
|
|
|
|
@property
|
|
|
|
def is_on(self):
|
|
|
|
"""Return true if switch is on."""
|
2019-07-31 19:25:30 +00:00
|
|
|
return self._switch.power == "ON"
|
2017-08-06 18:30:28 +00:00
|
|
|
|
|
|
|
def turn_on(self, **kwargs):
|
|
|
|
"""Instruct the switch to turn on & adjust brightness."""
|
2019-07-31 19:25:30 +00:00
|
|
|
attribs = {"power": "ON"}
|
2017-08-06 18:30:28 +00:00
|
|
|
|
|
|
|
if ATTR_BRIGHTNESS in kwargs:
|
2019-07-31 19:25:30 +00:00
|
|
|
min_level = self._switch.data.get("minLevel", 0)
|
|
|
|
max_level = self._switch.data.get("maxLevel", 100)
|
2017-08-06 18:30:28 +00:00
|
|
|
brightness = int(kwargs[ATTR_BRIGHTNESS] * max_level / 255)
|
|
|
|
brightness = max(brightness, min_level)
|
2019-07-31 19:25:30 +00:00
|
|
|
attribs["brightness"] = brightness
|
2017-08-06 18:30:28 +00:00
|
|
|
|
|
|
|
if ATTR_TRANSITION in kwargs:
|
|
|
|
transition = int(kwargs[ATTR_TRANSITION])
|
2019-07-31 19:25:30 +00:00
|
|
|
attribs["fadeOnTime"] = attribs["fadeOffTime"] = transition
|
2017-08-06 18:30:28 +00:00
|
|
|
|
|
|
|
try:
|
|
|
|
self._switch.update_attributes(attribs)
|
|
|
|
except ValueError:
|
2019-07-31 19:25:30 +00:00
|
|
|
_LOGGER.error("Failed to turn on myLeviton switch.")
|
2017-08-06 18:30:28 +00:00
|
|
|
|
|
|
|
def turn_off(self, **kwargs):
|
|
|
|
"""Instruct the switch to turn off."""
|
2019-07-31 19:25:30 +00:00
|
|
|
attribs = {"power": "OFF"}
|
2017-08-06 18:30:28 +00:00
|
|
|
try:
|
|
|
|
self._switch.update_attributes(attribs)
|
|
|
|
except ValueError:
|
2019-07-31 19:25:30 +00:00
|
|
|
_LOGGER.error("Failed to turn off myLeviton switch.")
|
2017-08-06 18:30:28 +00:00
|
|
|
|
|
|
|
def update(self):
|
|
|
|
"""Fetch new state data for this switch."""
|
|
|
|
try:
|
|
|
|
self._switch.refresh()
|
|
|
|
except ValueError:
|
2019-07-31 19:25:30 +00:00
|
|
|
_LOGGER.error("Failed to update myLeviton switch data.")
|