core/homeassistant/components/dlink/switch.py

178 lines
5.1 KiB
Python
Raw Normal View History

"""Support for D-Link W215 smart switch."""
from __future__ import annotations
2018-10-26 13:45:57 +00:00
from datetime import timedelta
import logging
import urllib
2019-12-05 05:22:37 +00:00
from pyW215.pyW215 import SmartPlug
2016-08-20 22:25:11 +00:00
import voluptuous as vol
from homeassistant.components.switch import PLATFORM_SCHEMA, SwitchEntity
2018-10-26 13:45:57 +00:00
from homeassistant.const import (
2019-07-31 19:25:30 +00:00
ATTR_TEMPERATURE,
CONF_HOST,
CONF_NAME,
CONF_PASSWORD,
CONF_USERNAME,
TEMP_CELSIUS,
)
from homeassistant.core import HomeAssistant
2016-08-20 22:25:11 +00:00
import homeassistant.helpers.config_validation as cv
from homeassistant.helpers.entity_platform import AddEntitiesCallback
from homeassistant.helpers.typing import ConfigType, DiscoveryInfoType
from homeassistant.util import dt as dt_util
_LOGGER = logging.getLogger(__name__)
2019-07-31 19:25:30 +00:00
ATTR_TOTAL_CONSUMPTION = "total_consumption"
2018-10-26 13:45:57 +00:00
2019-07-31 19:25:30 +00:00
CONF_USE_LEGACY_PROTOCOL = "use_legacy_protocol"
2018-10-26 13:45:57 +00:00
DEFAULT_NAME = "D-Link Smart Plug W215"
2019-07-31 19:25:30 +00:00
DEFAULT_PASSWORD = ""
DEFAULT_USERNAME = "admin"
2018-10-26 13:45:57 +00:00
SCAN_INTERVAL = timedelta(minutes=2)
2016-08-20 22:25:11 +00:00
2019-07-31 19:25:30 +00:00
PLATFORM_SCHEMA = PLATFORM_SCHEMA.extend(
{
vol.Required(CONF_HOST): cv.string,
vol.Required(CONF_PASSWORD, default=DEFAULT_PASSWORD): cv.string,
vol.Required(CONF_USERNAME, default=DEFAULT_USERNAME): cv.string,
vol.Optional(CONF_NAME, default=DEFAULT_NAME): cv.string,
vol.Optional(CONF_USE_LEGACY_PROTOCOL, default=False): cv.boolean,
}
)
2016-08-20 22:25:11 +00:00
def setup_platform(
hass: HomeAssistant,
config: ConfigType,
add_entities: AddEntitiesCallback,
discovery_info: DiscoveryInfoType | None = None,
) -> None:
"""Set up a D-Link Smart Plug."""
host = config[CONF_HOST]
username = config[CONF_USERNAME]
password = config[CONF_PASSWORD]
use_legacy_protocol = config[CONF_USE_LEGACY_PROTOCOL]
name = config[CONF_NAME]
2016-08-20 22:25:11 +00:00
2018-10-26 13:45:57 +00:00
smartplug = SmartPlug(host, password, username, use_legacy_protocol)
data = SmartPlugData(smartplug)
2016-11-17 02:55:58 +00:00
add_entities([SmartPlugSwitch(hass, data, name)], True)
class SmartPlugSwitch(SwitchEntity):
2018-10-26 13:45:57 +00:00
"""Representation of a D-Link Smart Plug switch."""
2016-03-08 12:35:39 +00:00
2016-11-17 02:55:58 +00:00
def __init__(self, hass, data, name):
2016-03-08 12:35:39 +00:00
"""Initialize the switch."""
self.units = hass.config.units
2016-11-17 02:55:58 +00:00
self.data = data
self._name = name
@property
def name(self):
2018-10-26 13:45:57 +00:00
"""Return the name of the Smart Plug."""
return self._name
@property
def extra_state_attributes(self):
"""Return the state attributes of the device."""
try:
2019-07-31 19:25:30 +00:00
ui_temp = self.units.temperature(int(self.data.temperature), TEMP_CELSIUS)
temperature = ui_temp
except (ValueError, TypeError):
temperature = None
try:
total_consumption = float(self.data.total_consumption)
except (ValueError, TypeError):
total_consumption = None
attrs = {
ATTR_TOTAL_CONSUMPTION: total_consumption,
ATTR_TEMPERATURE: temperature,
}
return attrs
@property
def current_power_w(self):
2016-03-08 12:35:39 +00:00
"""Return the current power usage in Watt."""
try:
2016-11-17 02:55:58 +00:00
return float(self.data.current_consumption)
except (ValueError, TypeError):
return None
@property
def is_on(self):
2016-03-08 12:35:39 +00:00
"""Return true if switch is on."""
2019-07-31 19:25:30 +00:00
return self.data.state == "ON"
def turn_on(self, **kwargs):
2016-03-08 12:35:39 +00:00
"""Turn the switch on."""
2019-07-31 19:25:30 +00:00
self.data.smartplug.state = "ON"
def turn_off(self, **kwargs):
2016-03-08 12:35:39 +00:00
"""Turn the switch off."""
2019-07-31 19:25:30 +00:00
self.data.smartplug.state = "OFF"
2016-11-17 02:55:58 +00:00
def update(self):
"""Get the latest data from the smart plug and updates the states."""
self.data.update()
@property
def available(self) -> bool:
"""Return True if entity is available."""
return self.data.available
2016-11-17 02:55:58 +00:00
class SmartPlugData:
2016-11-17 02:55:58 +00:00
"""Get the latest data from smart plug."""
def __init__(self, smartplug):
"""Initialize the data object."""
self.smartplug = smartplug
self.state = None
self.temperature = None
self.current_consumption = None
self.total_consumption = None
self.available = False
self._n_tried = 0
self._last_tried = None
2016-11-17 02:55:58 +00:00
def update(self):
"""Get the latest data from the smart plug."""
if self._last_tried is not None:
2019-07-31 19:25:30 +00:00
last_try_s = (dt_util.now() - self._last_tried).total_seconds() / 60
retry_seconds = min(self._n_tried * 2, 10) - last_try_s
if self._n_tried > 0 and retry_seconds > 0:
_LOGGER.warning("Waiting %s s to retry", retry_seconds)
return
_state = "unknown"
2018-10-26 13:45:57 +00:00
try:
self._last_tried = dt_util.now()
_state = self.smartplug.state
except urllib.error.HTTPError:
2018-10-26 13:45:57 +00:00
_LOGGER.error("D-Link connection problem")
if _state == "unknown":
self._n_tried += 1
self.available = False
2018-10-26 13:45:57 +00:00
_LOGGER.warning("Failed to connect to D-Link switch")
return
2018-10-26 13:45:57 +00:00
self.state = _state
self.available = True
2016-11-17 02:55:58 +00:00
self.temperature = self.smartplug.temperature
self.current_consumption = self.smartplug.current_consumption
self.total_consumption = self.smartplug.total_consumption
self._n_tried = 0