Compare commits
5 Commits
6.3.0.beta2
...
6.3.2
| Author | SHA1 | Date | |
|---|---|---|---|
| 668053b352 | |||
| 6ff9ff1ee5 | |||
| 3f95ed74f4 | |||
| 6e42904ddf | |||
| 4c1fc396fb |
@@ -13,7 +13,6 @@ from homeassistant.util import dt as dt_util
|
|||||||
from homeassistant.core import (
|
from homeassistant.core import (
|
||||||
HomeAssistant,
|
HomeAssistant,
|
||||||
callback,
|
callback,
|
||||||
CoreState,
|
|
||||||
Event,
|
Event,
|
||||||
State,
|
State,
|
||||||
)
|
)
|
||||||
@@ -57,7 +56,6 @@ from homeassistant.const import (
|
|||||||
STATE_UNKNOWN,
|
STATE_UNKNOWN,
|
||||||
STATE_OFF,
|
STATE_OFF,
|
||||||
STATE_ON,
|
STATE_ON,
|
||||||
EVENT_HOMEASSISTANT_START,
|
|
||||||
STATE_HOME,
|
STATE_HOME,
|
||||||
STATE_NOT_HOME,
|
STATE_NOT_HOME,
|
||||||
)
|
)
|
||||||
@@ -299,7 +297,7 @@ class BaseThermostat(ClimateEntity, RestoreEntity, Generic[T]):
|
|||||||
self._presets: dict[str, Any] = {} # presets
|
self._presets: dict[str, Any] = {} # presets
|
||||||
self._presets_away: dict[str, Any] = {} # presets_away
|
self._presets_away: dict[str, Any] = {} # presets_away
|
||||||
|
|
||||||
self._attr_preset_modes: list[str] | None
|
self._attr_preset_modes: list[str] = []
|
||||||
|
|
||||||
self._use_central_config_temperature = False
|
self._use_central_config_temperature = False
|
||||||
|
|
||||||
@@ -2183,7 +2181,7 @@ class BaseThermostat(ClimateEntity, RestoreEntity, Generic[T]):
|
|||||||
new_central_mode,
|
new_central_mode,
|
||||||
)
|
)
|
||||||
|
|
||||||
first_init = self._last_central_mode == None
|
first_init = self._last_central_mode is None
|
||||||
|
|
||||||
self._last_central_mode = new_central_mode
|
self._last_central_mode = new_central_mode
|
||||||
|
|
||||||
|
|||||||
@@ -99,30 +99,31 @@ class VersatileThermostatBaseConfigFlow(FlowHandler):
|
|||||||
|
|
||||||
def _init_feature_flags(self, _):
|
def _init_feature_flags(self, _):
|
||||||
"""Fix features selection depending to infos"""
|
"""Fix features selection depending to infos"""
|
||||||
is_empty: bool = False # TODO remove this not bool(infos)
|
|
||||||
is_central_config = (
|
is_central_config = (
|
||||||
self._infos.get(CONF_THERMOSTAT_TYPE) == CONF_THERMOSTAT_CENTRAL_CONFIG
|
self._infos.get(CONF_THERMOSTAT_TYPE) == CONF_THERMOSTAT_CENTRAL_CONFIG
|
||||||
)
|
)
|
||||||
|
|
||||||
self._infos[CONF_USE_WINDOW_FEATURE] = (
|
self._infos[CONF_USE_WINDOW_FEATURE] = (
|
||||||
is_empty
|
self._infos.get(CONF_USE_WINDOW_CENTRAL_CONFIG)
|
||||||
or self._infos.get(CONF_WINDOW_SENSOR) is not None
|
or self._infos.get(CONF_WINDOW_SENSOR) is not None
|
||||||
or self._infos.get(CONF_WINDOW_AUTO_OPEN_THRESHOLD) is not None
|
or self._infos.get(CONF_WINDOW_AUTO_OPEN_THRESHOLD) is not None
|
||||||
)
|
)
|
||||||
self._infos[CONF_USE_MOTION_FEATURE] = (
|
self._infos[CONF_USE_MOTION_FEATURE] = self._infos.get(
|
||||||
is_empty
|
CONF_USE_MOTION_FEATURE
|
||||||
or self._infos.get(CONF_MOTION_SENSOR) is not None
|
) and (self._infos.get(CONF_MOTION_SENSOR) is not None or is_central_config)
|
||||||
or is_central_config
|
|
||||||
)
|
self._infos[CONF_USE_POWER_FEATURE] = self._infos.get(
|
||||||
self._infos[CONF_USE_POWER_FEATURE] = is_empty or (
|
CONF_USE_POWER_CENTRAL_CONFIG
|
||||||
|
) or (
|
||||||
self._infos.get(CONF_POWER_SENSOR) is not None
|
self._infos.get(CONF_POWER_SENSOR) is not None
|
||||||
and self._infos.get(CONF_MAX_POWER_SENSOR) is not None
|
and self._infos.get(CONF_MAX_POWER_SENSOR) is not None
|
||||||
)
|
)
|
||||||
self._infos[CONF_USE_PRESENCE_FEATURE] = (
|
self._infos[CONF_USE_PRESENCE_FEATURE] = (
|
||||||
is_empty or self._infos.get(CONF_PRESENCE_SENSOR) is not None
|
self._infos.get(CONF_USE_PRESENCE_CENTRAL_CONFIG)
|
||||||
|
or self._infos.get(CONF_PRESENCE_SENSOR) is not None
|
||||||
)
|
)
|
||||||
|
|
||||||
self._infos[CONF_USE_CENTRAL_BOILER_FEATURE] = is_empty or (
|
self._infos[CONF_USE_CENTRAL_BOILER_FEATURE] = (
|
||||||
self._infos.get(CONF_CENTRAL_BOILER_ACTIVATION_SRV) is not None
|
self._infos.get(CONF_CENTRAL_BOILER_ACTIVATION_SRV) is not None
|
||||||
and self._infos.get(CONF_CENTRAL_BOILER_DEACTIVATION_SRV) is not None
|
and self._infos.get(CONF_CENTRAL_BOILER_DEACTIVATION_SRV) is not None
|
||||||
)
|
)
|
||||||
|
|||||||
@@ -142,7 +142,17 @@ class ThermostatOverClimate(BaseThermostat[UnderlyingClimate]):
|
|||||||
"""Sends the regulated temperature to all underlying"""
|
"""Sends the regulated temperature to all underlying"""
|
||||||
|
|
||||||
if self.hvac_mode == HVACMode.OFF:
|
if self.hvac_mode == HVACMode.OFF:
|
||||||
_LOGGER.debug("%s - don't send regulated temperature cause VTherm is off ")
|
_LOGGER.debug(
|
||||||
|
"%s - don't send regulated temperature cause VTherm is off ", self
|
||||||
|
)
|
||||||
|
return
|
||||||
|
|
||||||
|
if self.target_temperature is None:
|
||||||
|
_LOGGER.warning(
|
||||||
|
"%s - don't send regulated temperature cause VTherm target_temp (%s) is None. This should be a temporary warning message.",
|
||||||
|
self,
|
||||||
|
self.target_temperature,
|
||||||
|
)
|
||||||
return
|
return
|
||||||
|
|
||||||
_LOGGER.info(
|
_LOGGER.info(
|
||||||
@@ -169,16 +179,17 @@ class ThermostatOverClimate(BaseThermostat[UnderlyingClimate]):
|
|||||||
|
|
||||||
# use _attr_target_temperature_step to round value if _auto_regulation_dtemp is equal to 0
|
# use _attr_target_temperature_step to round value if _auto_regulation_dtemp is equal to 0
|
||||||
regulation_step = self._auto_regulation_dtemp if self._auto_regulation_dtemp else self._attr_target_temperature_step
|
regulation_step = self._auto_regulation_dtemp if self._auto_regulation_dtemp else self._attr_target_temperature_step
|
||||||
_LOGGER.debug("%s - usage of regulation_step: %.2f ",
|
_LOGGER.debug("%s - usage regulation_step: %.2f ", self, regulation_step)
|
||||||
self,
|
|
||||||
regulation_step)
|
if self.current_temperature is not None:
|
||||||
|
new_regulated_temp = round_to_nearest(
|
||||||
new_regulated_temp = round_to_nearest(
|
self._regulation_algo.calculate_regulated_temperature(
|
||||||
self._regulation_algo.calculate_regulated_temperature(
|
self.current_temperature, self._cur_ext_temp
|
||||||
self.current_temperature, self._cur_ext_temp
|
),
|
||||||
),
|
regulation_step,
|
||||||
regulation_step,
|
)
|
||||||
)
|
else:
|
||||||
|
new_regulated_temp = self.target_temperature
|
||||||
dtemp = new_regulated_temp - self._regulated_target_temp
|
dtemp = new_regulated_temp - self._regulated_target_temp
|
||||||
|
|
||||||
if not force and abs(dtemp) < self._auto_regulation_dtemp:
|
if not force and abs(dtemp) < self._auto_regulation_dtemp:
|
||||||
@@ -203,8 +214,10 @@ class ThermostatOverClimate(BaseThermostat[UnderlyingClimate]):
|
|||||||
offset_temp = 0
|
offset_temp = 0
|
||||||
device_temp = 0
|
device_temp = 0
|
||||||
if (
|
if (
|
||||||
|
# current_temperature is set
|
||||||
|
self.current_temperature is not None
|
||||||
# regulation can use the device_temp
|
# regulation can use the device_temp
|
||||||
self.auto_regulation_use_device_temp
|
and self.auto_regulation_use_device_temp
|
||||||
# and we have access to the device temp
|
# and we have access to the device temp
|
||||||
and (device_temp := under.underlying_current_temperature) is not None
|
and (device_temp := under.underlying_current_temperature) is not None
|
||||||
# and target is not reach (ie we need regulation)
|
# and target is not reach (ie we need regulation)
|
||||||
|
|||||||
@@ -12,6 +12,14 @@ from homeassistant.components.climate import (
|
|||||||
SERVICE_SET_TEMPERATURE,
|
SERVICE_SET_TEMPERATURE,
|
||||||
)
|
)
|
||||||
|
|
||||||
|
from custom_components.versatile_thermostat.config_flow import (
|
||||||
|
VersatileThermostatBaseConfigFlow,
|
||||||
|
)
|
||||||
|
from custom_components.versatile_thermostat.thermostat_valve import ThermostatOverValve
|
||||||
|
from custom_components.versatile_thermostat.thermostat_switch import (
|
||||||
|
ThermostatOverSwitch,
|
||||||
|
)
|
||||||
|
|
||||||
from .commons import *
|
from .commons import *
|
||||||
|
|
||||||
logging.getLogger().setLevel(logging.DEBUG)
|
logging.getLogger().setLevel(logging.DEBUG)
|
||||||
@@ -1013,3 +1021,72 @@ async def test_bug_508(
|
|||||||
),
|
),
|
||||||
]
|
]
|
||||||
)
|
)
|
||||||
|
|
||||||
|
|
||||||
|
@pytest.mark.parametrize("expected_lingering_tasks", [True])
|
||||||
|
@pytest.mark.parametrize("expected_lingering_timers", [True])
|
||||||
|
async def test_bug_500_1(hass: HomeAssistant, init_vtherm_api) -> None:
|
||||||
|
"""Test that the form is served with no input"""
|
||||||
|
|
||||||
|
config = {
|
||||||
|
CONF_THERMOSTAT_TYPE: CONF_THERMOSTAT_SWITCH,
|
||||||
|
CONF_USE_WINDOW_CENTRAL_CONFIG: True,
|
||||||
|
CONF_USE_POWER_CENTRAL_CONFIG: True,
|
||||||
|
CONF_USE_PRESENCE_CENTRAL_CONFIG: True,
|
||||||
|
CONF_USE_MOTION_FEATURE: True,
|
||||||
|
CONF_MOTION_SENSOR: "sensor.theMotionSensor",
|
||||||
|
}
|
||||||
|
|
||||||
|
flow = VersatileThermostatBaseConfigFlow(config)
|
||||||
|
|
||||||
|
assert flow._infos[CONF_USE_WINDOW_FEATURE] is True
|
||||||
|
assert flow._infos[CONF_USE_POWER_FEATURE] is True
|
||||||
|
assert flow._infos[CONF_USE_PRESENCE_FEATURE] is True
|
||||||
|
assert flow._infos[CONF_USE_MOTION_FEATURE] is True
|
||||||
|
|
||||||
|
|
||||||
|
@pytest.mark.parametrize("expected_lingering_tasks", [True])
|
||||||
|
@pytest.mark.parametrize("expected_lingering_timers", [True])
|
||||||
|
async def test_bug_500_2(hass: HomeAssistant, init_vtherm_api) -> None:
|
||||||
|
"""Test that the form is served with no input"""
|
||||||
|
|
||||||
|
config = {
|
||||||
|
CONF_THERMOSTAT_TYPE: CONF_THERMOSTAT_SWITCH,
|
||||||
|
CONF_USE_WINDOW_CENTRAL_CONFIG: False,
|
||||||
|
CONF_USE_POWER_CENTRAL_CONFIG: False,
|
||||||
|
CONF_USE_PRESENCE_CENTRAL_CONFIG: False,
|
||||||
|
CONF_USE_MOTION_FEATURE: False,
|
||||||
|
}
|
||||||
|
|
||||||
|
flow = VersatileThermostatBaseConfigFlow(config)
|
||||||
|
|
||||||
|
assert flow._infos[CONF_USE_WINDOW_FEATURE] is False
|
||||||
|
assert flow._infos[CONF_USE_POWER_FEATURE] is False
|
||||||
|
assert flow._infos[CONF_USE_PRESENCE_FEATURE] is False
|
||||||
|
assert flow._infos[CONF_USE_MOTION_FEATURE] is False
|
||||||
|
|
||||||
|
|
||||||
|
@pytest.mark.parametrize("expected_lingering_tasks", [True])
|
||||||
|
@pytest.mark.parametrize("expected_lingering_timers", [True])
|
||||||
|
async def test_bug_500_3(hass: HomeAssistant, init_vtherm_api) -> None:
|
||||||
|
"""Test that the form is served with no input"""
|
||||||
|
|
||||||
|
config = {
|
||||||
|
CONF_THERMOSTAT_TYPE: CONF_THERMOSTAT_SWITCH,
|
||||||
|
CONF_USE_WINDOW_CENTRAL_CONFIG: False,
|
||||||
|
CONF_WINDOW_SENSOR: "sensor.theWindowSensor",
|
||||||
|
CONF_USE_POWER_CENTRAL_CONFIG: False,
|
||||||
|
CONF_POWER_SENSOR: "sensor.thePowerSensor",
|
||||||
|
CONF_MAX_POWER_SENSOR: "sensor.theMaxPowerSensor",
|
||||||
|
CONF_USE_PRESENCE_CENTRAL_CONFIG: False,
|
||||||
|
CONF_PRESENCE_SENSOR: "sensor.thePresenceSensor",
|
||||||
|
CONF_USE_MOTION_FEATURE: True, # motion sensor need to be checked AND a motion sensor set
|
||||||
|
CONF_MOTION_SENSOR: "sensor.theMotionSensor",
|
||||||
|
}
|
||||||
|
|
||||||
|
flow = VersatileThermostatBaseConfigFlow(config)
|
||||||
|
|
||||||
|
assert flow._infos[CONF_USE_WINDOW_FEATURE] is True
|
||||||
|
assert flow._infos[CONF_USE_POWER_FEATURE] is True
|
||||||
|
assert flow._infos[CONF_USE_PRESENCE_FEATURE] is True
|
||||||
|
assert flow._infos[CONF_USE_MOTION_FEATURE] is True
|
||||||
|
|||||||
Reference in New Issue
Block a user