2019-02-13 20:21:14 +00:00
|
|
|
"""Support for Rheem EcoNet water heaters."""
|
2017-12-29 18:05:58 +00:00
|
|
|
import datetime
|
|
|
|
import logging
|
|
|
|
|
2019-11-25 15:06:37 +00:00
|
|
|
from pyeconet.api import PyEcoNet
|
2017-12-29 18:05:58 +00:00
|
|
|
import voluptuous as vol
|
|
|
|
|
2018-10-13 21:16:44 +00:00
|
|
|
from homeassistant.components.water_heater import (
|
2019-07-31 19:25:30 +00:00
|
|
|
PLATFORM_SCHEMA,
|
|
|
|
STATE_ECO,
|
|
|
|
STATE_ELECTRIC,
|
|
|
|
STATE_GAS,
|
|
|
|
STATE_HEAT_PUMP,
|
|
|
|
STATE_HIGH_DEMAND,
|
|
|
|
STATE_OFF,
|
|
|
|
STATE_PERFORMANCE,
|
|
|
|
SUPPORT_OPERATION_MODE,
|
|
|
|
SUPPORT_TARGET_TEMPERATURE,
|
2020-05-01 14:29:14 +00:00
|
|
|
WaterHeaterEntity,
|
2019-07-31 19:25:30 +00:00
|
|
|
)
|
2018-01-21 06:35:38 +00:00
|
|
|
from homeassistant.const import (
|
2019-07-31 19:25:30 +00:00
|
|
|
ATTR_ENTITY_ID,
|
|
|
|
ATTR_TEMPERATURE,
|
|
|
|
CONF_PASSWORD,
|
|
|
|
CONF_USERNAME,
|
|
|
|
TEMP_FAHRENHEIT,
|
|
|
|
)
|
2017-12-29 18:05:58 +00:00
|
|
|
import homeassistant.helpers.config_validation as cv
|
|
|
|
|
2019-11-27 10:18:21 +00:00
|
|
|
from .const import DOMAIN, SERVICE_ADD_VACATION, SERVICE_DELETE_VACATION
|
|
|
|
|
2017-12-29 18:05:58 +00:00
|
|
|
_LOGGER = logging.getLogger(__name__)
|
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
ATTR_VACATION_START = "next_vacation_start_date"
|
|
|
|
ATTR_VACATION_END = "next_vacation_end_date"
|
|
|
|
ATTR_ON_VACATION = "on_vacation"
|
|
|
|
ATTR_TODAYS_ENERGY_USAGE = "todays_energy_usage"
|
|
|
|
ATTR_IN_USE = "in_use"
|
2017-12-29 18:05:58 +00:00
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
ATTR_START_DATE = "start_date"
|
|
|
|
ATTR_END_DATE = "end_date"
|
2017-12-29 18:05:58 +00:00
|
|
|
|
2020-04-10 22:37:43 +00:00
|
|
|
ATTR_LOWER_TEMP = "lower_temp"
|
|
|
|
ATTR_UPPER_TEMP = "upper_temp"
|
|
|
|
ATTR_IS_ENABLED = "is_enabled"
|
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
SUPPORT_FLAGS_HEATER = SUPPORT_TARGET_TEMPERATURE | SUPPORT_OPERATION_MODE
|
2017-12-29 18:05:58 +00:00
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
ADD_VACATION_SCHEMA = vol.Schema(
|
|
|
|
{
|
|
|
|
vol.Optional(ATTR_ENTITY_ID): cv.entity_ids,
|
|
|
|
vol.Optional(ATTR_START_DATE): cv.positive_int,
|
|
|
|
vol.Required(ATTR_END_DATE): cv.positive_int,
|
|
|
|
}
|
|
|
|
)
|
2017-12-29 18:05:58 +00:00
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
DELETE_VACATION_SCHEMA = vol.Schema({vol.Optional(ATTR_ENTITY_ID): cv.entity_ids})
|
2017-12-29 18:05:58 +00:00
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
ECONET_DATA = "econet"
|
2017-12-29 18:05:58 +00:00
|
|
|
|
2019-02-26 22:49:15 +00:00
|
|
|
ECONET_STATE_TO_HA = {
|
2019-07-31 19:25:30 +00:00
|
|
|
"Energy Saver": STATE_ECO,
|
|
|
|
"gas": STATE_GAS,
|
|
|
|
"High Demand": STATE_HIGH_DEMAND,
|
|
|
|
"Off": STATE_OFF,
|
|
|
|
"Performance": STATE_PERFORMANCE,
|
|
|
|
"Heat Pump Only": STATE_HEAT_PUMP,
|
|
|
|
"Electric-Only": STATE_ELECTRIC,
|
|
|
|
"Electric": STATE_ELECTRIC,
|
|
|
|
"Heat Pump": STATE_HEAT_PUMP,
|
2017-12-29 18:05:58 +00:00
|
|
|
}
|
|
|
|
|
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-12-29 18:05:58 +00:00
|
|
|
|
|
|
|
|
2018-08-24 14:37:30 +00:00
|
|
|
def setup_platform(hass, config, add_entities, discovery_info=None):
|
2017-12-29 18:05:58 +00:00
|
|
|
"""Set up the EcoNet water heaters."""
|
|
|
|
|
|
|
|
hass.data[ECONET_DATA] = {}
|
2019-07-31 19:25:30 +00:00
|
|
|
hass.data[ECONET_DATA]["water_heaters"] = []
|
2017-12-29 18:05:58 +00:00
|
|
|
|
|
|
|
username = config.get(CONF_USERNAME)
|
|
|
|
password = config.get(CONF_PASSWORD)
|
|
|
|
|
|
|
|
econet = PyEcoNet(username, password)
|
|
|
|
water_heaters = econet.get_water_heaters()
|
|
|
|
hass_water_heaters = [
|
2019-07-31 19:25:30 +00:00
|
|
|
EcoNetWaterHeater(water_heater) for water_heater in water_heaters
|
|
|
|
]
|
2018-08-24 14:37:30 +00:00
|
|
|
add_entities(hass_water_heaters)
|
2019-07-31 19:25:30 +00:00
|
|
|
hass.data[ECONET_DATA]["water_heaters"].extend(hass_water_heaters)
|
2017-12-29 18:05:58 +00:00
|
|
|
|
|
|
|
def service_handle(service):
|
2018-01-21 06:35:38 +00:00
|
|
|
"""Handle the service calls."""
|
2019-07-31 19:25:30 +00:00
|
|
|
entity_ids = service.data.get("entity_id")
|
|
|
|
all_heaters = hass.data[ECONET_DATA]["water_heaters"]
|
2017-12-29 18:05:58 +00:00
|
|
|
_heaters = [
|
2019-07-31 19:25:30 +00:00
|
|
|
x for x in all_heaters if not entity_ids or x.entity_id in entity_ids
|
|
|
|
]
|
2017-12-29 18:05:58 +00:00
|
|
|
|
|
|
|
for _water_heater in _heaters:
|
|
|
|
if service.service == SERVICE_ADD_VACATION:
|
|
|
|
start = service.data.get(ATTR_START_DATE)
|
|
|
|
end = service.data.get(ATTR_END_DATE)
|
|
|
|
_water_heater.add_vacation(start, end)
|
|
|
|
if service.service == SERVICE_DELETE_VACATION:
|
|
|
|
for vacation in _water_heater.water_heater.vacations:
|
|
|
|
vacation.delete()
|
|
|
|
|
|
|
|
_water_heater.schedule_update_ha_state(True)
|
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
hass.services.register(
|
|
|
|
DOMAIN, SERVICE_ADD_VACATION, service_handle, schema=ADD_VACATION_SCHEMA
|
|
|
|
)
|
2017-12-29 18:05:58 +00:00
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
hass.services.register(
|
|
|
|
DOMAIN, SERVICE_DELETE_VACATION, service_handle, schema=DELETE_VACATION_SCHEMA
|
|
|
|
)
|
2017-12-29 18:05:58 +00:00
|
|
|
|
|
|
|
|
2020-05-01 14:29:14 +00:00
|
|
|
class EcoNetWaterHeater(WaterHeaterEntity):
|
2017-12-29 18:05:58 +00:00
|
|
|
"""Representation of an EcoNet water heater."""
|
|
|
|
|
|
|
|
def __init__(self, water_heater):
|
|
|
|
"""Initialize the water heater."""
|
|
|
|
self.water_heater = water_heater
|
2019-02-26 22:49:15 +00:00
|
|
|
self.supported_modes = self.water_heater.supported_modes
|
|
|
|
self.econet_state_to_ha = {}
|
2019-02-27 02:50:34 +00:00
|
|
|
self.ha_state_to_econet = {}
|
|
|
|
for mode in ECONET_STATE_TO_HA:
|
2019-02-26 22:49:15 +00:00
|
|
|
if mode in self.supported_modes:
|
|
|
|
self.econet_state_to_ha[mode] = ECONET_STATE_TO_HA.get(mode)
|
2019-02-27 02:50:34 +00:00
|
|
|
for key, value in self.econet_state_to_ha.items():
|
2019-02-26 22:49:15 +00:00
|
|
|
self.ha_state_to_econet[value] = key
|
|
|
|
for mode in self.supported_modes:
|
|
|
|
if mode not in ECONET_STATE_TO_HA:
|
2020-04-04 23:32:58 +00:00
|
|
|
error = f"Invalid operation mode mapping. {mode} doesn't map. Please report this."
|
2019-02-26 22:49:15 +00:00
|
|
|
_LOGGER.error(error)
|
2017-12-29 18:05:58 +00:00
|
|
|
|
|
|
|
@property
|
|
|
|
def name(self):
|
|
|
|
"""Return the device name."""
|
|
|
|
return self.water_heater.name
|
|
|
|
|
|
|
|
@property
|
|
|
|
def available(self):
|
|
|
|
"""Return if the the device is online or not."""
|
|
|
|
return self.water_heater.is_connected
|
|
|
|
|
|
|
|
@property
|
|
|
|
def temperature_unit(self):
|
|
|
|
"""Return the unit of measurement."""
|
|
|
|
return TEMP_FAHRENHEIT
|
|
|
|
|
|
|
|
@property
|
|
|
|
def device_state_attributes(self):
|
2018-01-21 06:35:38 +00:00
|
|
|
"""Return the optional device state attributes."""
|
2017-12-29 18:05:58 +00:00
|
|
|
data = {}
|
|
|
|
vacations = self.water_heater.get_vacations()
|
|
|
|
if vacations:
|
|
|
|
data[ATTR_VACATION_START] = vacations[0].start_date
|
|
|
|
data[ATTR_VACATION_END] = vacations[0].end_date
|
|
|
|
data[ATTR_ON_VACATION] = self.water_heater.is_on_vacation
|
|
|
|
todays_usage = self.water_heater.total_usage_for_today
|
|
|
|
if todays_usage:
|
|
|
|
data[ATTR_TODAYS_ENERGY_USAGE] = todays_usage
|
|
|
|
data[ATTR_IN_USE] = self.water_heater.in_use
|
|
|
|
|
2020-04-10 22:37:43 +00:00
|
|
|
if self.water_heater.lower_temp is not None:
|
|
|
|
data[ATTR_LOWER_TEMP] = round(self.water_heater.lower_temp, 2)
|
|
|
|
if self.water_heater.upper_temp is not None:
|
|
|
|
data[ATTR_UPPER_TEMP] = round(self.water_heater.upper_temp, 2)
|
|
|
|
if self.water_heater.is_enabled is not None:
|
|
|
|
data[ATTR_IS_ENABLED] = self.water_heater.is_enabled
|
|
|
|
|
2017-12-29 18:05:58 +00:00
|
|
|
return data
|
|
|
|
|
|
|
|
@property
|
|
|
|
def current_operation(self):
|
|
|
|
"""
|
|
|
|
Return current operation as one of the following.
|
|
|
|
|
2018-01-21 06:35:38 +00:00
|
|
|
["eco", "heat_pump", "high_demand", "electric_only"]
|
2017-12-29 18:05:58 +00:00
|
|
|
"""
|
2019-02-26 22:49:15 +00:00
|
|
|
current_op = self.econet_state_to_ha.get(self.water_heater.mode)
|
2017-12-29 18:05:58 +00:00
|
|
|
return current_op
|
|
|
|
|
|
|
|
@property
|
|
|
|
def operation_list(self):
|
|
|
|
"""List of available operation modes."""
|
|
|
|
op_list = []
|
2019-02-26 22:49:15 +00:00
|
|
|
for mode in self.supported_modes:
|
|
|
|
ha_mode = self.econet_state_to_ha.get(mode)
|
2017-12-29 18:05:58 +00:00
|
|
|
if ha_mode is not None:
|
|
|
|
op_list.append(ha_mode)
|
|
|
|
return op_list
|
|
|
|
|
|
|
|
@property
|
|
|
|
def supported_features(self):
|
|
|
|
"""Return the list of supported features."""
|
|
|
|
return SUPPORT_FLAGS_HEATER
|
|
|
|
|
|
|
|
def set_temperature(self, **kwargs):
|
|
|
|
"""Set new target temperature."""
|
|
|
|
target_temp = kwargs.get(ATTR_TEMPERATURE)
|
|
|
|
if target_temp is not None:
|
|
|
|
self.water_heater.set_target_set_point(target_temp)
|
|
|
|
else:
|
2018-01-21 06:35:38 +00:00
|
|
|
_LOGGER.error("A target temperature must be provided")
|
2017-12-29 18:05:58 +00:00
|
|
|
|
|
|
|
def set_operation_mode(self, operation_mode):
|
|
|
|
"""Set operation mode."""
|
2019-02-26 22:49:15 +00:00
|
|
|
op_mode_to_set = self.ha_state_to_econet.get(operation_mode)
|
2017-12-29 18:05:58 +00:00
|
|
|
if op_mode_to_set is not None:
|
|
|
|
self.water_heater.set_mode(op_mode_to_set)
|
|
|
|
else:
|
2018-01-21 06:35:38 +00:00
|
|
|
_LOGGER.error("An operation mode must be provided")
|
2017-12-29 18:05:58 +00:00
|
|
|
|
|
|
|
def add_vacation(self, start, end):
|
|
|
|
"""Add a vacation to this water heater."""
|
|
|
|
if not start:
|
|
|
|
start = datetime.datetime.now()
|
|
|
|
else:
|
|
|
|
start = datetime.datetime.fromtimestamp(start)
|
|
|
|
end = datetime.datetime.fromtimestamp(end)
|
|
|
|
self.water_heater.set_vacation_mode(start, end)
|
|
|
|
|
|
|
|
def update(self):
|
|
|
|
"""Get the latest date."""
|
|
|
|
self.water_heater.update_state()
|
|
|
|
|
|
|
|
@property
|
|
|
|
def target_temperature(self):
|
|
|
|
"""Return the temperature we try to reach."""
|
|
|
|
return self.water_heater.set_point
|
|
|
|
|
|
|
|
@property
|
|
|
|
def min_temp(self):
|
|
|
|
"""Return the minimum temperature."""
|
|
|
|
return self.water_heater.min_set_point
|
|
|
|
|
|
|
|
@property
|
|
|
|
def max_temp(self):
|
|
|
|
"""Return the maximum temperature."""
|
|
|
|
return self.water_heater.max_set_point
|