* Climate 1.0 / part 1/2/3 * fix flake * Lint * Update Google Assistant * ambiclimate to climate 1.0 (#24911) * Fix Alexa * Lint * Migrate zhong_hong * Migrate tuya * Migrate honeywell to new climate schema (#24257) * Update one * Fix model climate v2 * Cleanup p4 * Add comfort hold mode * Fix old code * Update homeassistant/components/climate/__init__.py Co-Authored-By: Paulus Schoutsen <paulus@home-assistant.io> * Update homeassistant/components/climate/const.py Co-Authored-By: Paulus Schoutsen <paulus@home-assistant.io> * First renaming * Rename operation to hvac for paulus * Rename hold mode to preset mode * Cleanup & update comments * Remove on/off * Fix supported feature count * Update services * Update demo * Fix tests & use current_hvac * Update comment * Fix tests & add typing * Add more typing * Update modes * Fix tests * Cleanup low/high with range * Update homematic part 1 * Finish homematic * Fix lint * fix hm mapping * Support simple devices * convert lcn * migrate oem * Fix xs1 * update hive * update mil * Update toon * migrate deconz * cleanup * update tesla * Fix lint * Fix vera * Migrate zwave * Migrate velbus * Cleanup humity feature * Cleanup * Migrate wink * migrate dyson * Fix current hvac * Renaming * Fix lint * Migrate tfiac * migrate tado * Fix PRESET can be None * apply PR#23913 from dev * remove EU component, etc. * remove EU component, etc. * ready to test now * de-linted * some tweaks * de-lint * better handling of edge cases * delint * fix set_mode typos * apply PR#23913 from dev * remove EU component, etc. * ready to test now * de-linted * some tweaks * de-lint * better handling of edge cases * delint * fix set_mode typos * delint, move debug code * away preset now working * code tidy-up * code tidy-up 2 * code tidy-up 3 * address issues #18932, #15063 * address issues #18932, #15063 - 2/2 * refactor MODE_AUTO to MODE_HEAT_COOL and use F not C * add low/high to set_temp * add low/high to set_temp 2 * add low/high to set_temp - delint * run HA scripts * port changes from PR #24402 * manual rebase * manual rebase 2 * delint * minor change * remove SUPPORT_HVAC_ACTION * Migrate radiotherm * Convert touchline * Migrate flexit * Migrate nuheat * Migrate maxcube * Fix names maxcube const * Migrate proliphix * Migrate heatmiser * Migrate fritzbox * Migrate opentherm_gw * Migrate venstar * Migrate daikin * Migrate modbus * Fix elif * Migrate Homematic IP Cloud to climate-1.0 (#24913) * hmip climate fix * Update hvac_mode and preset_mode * fix lint * Fix lint * Migrate generic_thermostat * Migrate incomfort to new climate schema (#24915) * initial commit * Update climate.py * Migrate eq3btsmart * Lint * cleanup PRESET_MANUAL * Migrate ecobee * No conditional features * KNX: Migrate climate component to new climate platform (#24931) * Migrate climate component * Remove unused code * Corrected line length * Lint * Lint * fix tests * Fix value * Migrate geniushub to new climate schema (#24191) * Update one * Fix model climate v2 * Cleanup p4 * Add comfort hold mode * Fix old code * Update homeassistant/components/climate/__init__.py Co-Authored-By: Paulus Schoutsen <paulus@home-assistant.io> * Update homeassistant/components/climate/const.py Co-Authored-By: Paulus Schoutsen <paulus@home-assistant.io> * First renaming * Rename operation to hvac for paulus * Rename hold mode to preset mode * Cleanup & update comments * Remove on/off * Fix supported feature count * Update services * Update demo * Fix tests & use current_hvac * Update comment * Fix tests & add typing * Add more typing * Update modes * Fix tests * Cleanup low/high with range * Update homematic part 1 * Finish homematic * Fix lint * fix hm mapping * Support simple devices * convert lcn * migrate oem * Fix xs1 * update hive * update mil * Update toon * migrate deconz * cleanup * update tesla * Fix lint * Fix vera * Migrate zwave * Migrate velbus * Cleanup humity feature * Cleanup * Migrate wink * migrate dyson * Fix current hvac * Renaming * Fix lint * Migrate tfiac * migrate tado * delinted * delinted * use latest client * clean up mappings * clean up mappings * add duration to set_temperature * add duration to set_temperature * manual rebase * tweak * fix regression * small fix * fix rebase mixup * address comments * finish refactor * fix regression * tweak type hints * delint * manual rebase * WIP: Fixes for honeywell migration to climate-1.0 (#24938) * add type hints * code tidy-up * Fixes for incomfort migration to climate-1.0 (#24936) * delint type hints * no async unless await * revert: no async unless await * revert: no async unless await 2 * delint * fix typo * Fix homekit_controller on climate-1.0 (#24948) * Fix tests on climate-1.0 branch * As part of climate-1.0, make state return the heating-cooling.current characteristic * Fixes from review * lint * Fix imports * Migrate stibel_eltron * Fix lint * Migrate coolmaster to climate 1.0 (#24967) * Migrate coolmaster to climate 1.0 * fix lint errors * More lint fixes * Fix demo to work with UI * Migrate spider * Demo update * Updated frontend to 20190705.0 * Fix boost mode (#24980) * Prepare Netatmo for climate 1.0 (#24973) * Migration Netatmo * Address comments * Update climate.py * Migrate ephember * Migrate Sensibo * Implemented review comments (#24942) * Migrate ESPHome * Migrate MQTT * Migrate Nest * Migrate melissa * Initial/partial migration of ST * Migrate ST * Remove Away mode (#24995) * Migrate evohome, cache access tokens (#24491) * add water_heater, add storage - initial commit * add water_heater, add storage - initial commit delint add missing code desiderata update honeywell client library & CODEOWNER add auth_tokens code, refactor & delint refactor for broker delint * Add Broker, Water Heater & Refactor add missing code desiderata * update honeywell client library & CODEOWNER add auth_tokens code, refactor & delint refactor for broker * bugfix - loc_idx may not be 0 more refactor - ensure pure async more refactoring appears all r/o attributes are working tweak precsion, DHW & delint remove unused code remove unused code 2 remove unused code, refactor _save_auth_tokens() * support RoundThermostat bugfix opmode, switch to util.dt, add until=1h revert breaking change * store at_expires as naive UTC remove debug code delint tidy up exception handling delint add water_heater, add storage - initial commit delint add missing code desiderata update honeywell client library & CODEOWNER add auth_tokens code, refactor & delint refactor for broker add water_heater, add storage - initial commit delint add missing code desiderata update honeywell client library & CODEOWNER add auth_tokens code, refactor & delint refactor for broker delint bugfix - loc_idx may not be 0 more refactor - ensure pure async more refactoring appears all r/o attributes are working tweak precsion, DHW & delint remove unused code remove unused code 2 remove unused code, refactor _save_auth_tokens() support RoundThermostat bugfix opmode, switch to util.dt, add until=1h revert breaking change store at_expires as naive UTC remove debug code delint tidy up exception handling delint * update CODEOWNERS * fix regression * fix requirements * migrate to climate-1.0 * tweaking * de-lint * TCS working? & delint * tweaking * TCS code finalised * remove available() logic * refactor _switchpoints() * tidy up switchpoint code * tweak * teaking device_state_attributes * some refactoring * move PRESET_CUSTOM back to evohome * move CONF_ACCESS_TOKEN_EXPIRES CONF_REFRESH_TOKEN back to evohome * refactor SP code and dt conversion * delinted * delinted * remove water_heater * fix regression * Migrate homekit * Cleanup away mode * Fix tests * add helpers * fix tests melissa * Fix nehueat * fix zwave * add more tests * fix deconz * Fix climate test emulate_hue * fix tests * fix dyson tests * fix demo with new layout * fix honeywell * Switch homekit_controller to use HVAC_MODE_HEAT_COOL instead of HVAC_MODE_AUTO (#25009) * Lint * PyLint * Pylint * fix fritzbox tests * Fix google * Fix all tests * Fix lint * Fix auto for homekit like controler * Fix lint * fix lint
200 lines
6.3 KiB
Python
200 lines
6.3 KiB
Python
"""deCONZ climate platform tests."""
|
|
from copy import deepcopy
|
|
from unittest.mock import Mock, patch
|
|
|
|
import asynctest
|
|
|
|
from homeassistant import config_entries
|
|
from homeassistant.components import deconz
|
|
from homeassistant.helpers.dispatcher import async_dispatcher_send
|
|
from homeassistant.setup import async_setup_component
|
|
|
|
import homeassistant.components.climate as climate
|
|
|
|
from tests.common import mock_coro
|
|
|
|
|
|
SENSOR = {
|
|
"1": {
|
|
"id": "Climate 1 id",
|
|
"name": "Climate 1 name",
|
|
"type": "ZHAThermostat",
|
|
"state": {"on": True, "temperature": 2260, "valve": 30},
|
|
"config": {"battery": 100, "heatsetpoint": 2200, "mode": "auto",
|
|
"offset": 10, "reachable": True},
|
|
"uniqueid": "00:00:00:00:00:00:00:00-00"
|
|
},
|
|
"2": {
|
|
"id": "Sensor 2 id",
|
|
"name": "Sensor 2 name",
|
|
"type": "ZHAPresence",
|
|
"state": {"presence": False},
|
|
"config": {}
|
|
}
|
|
}
|
|
|
|
ENTRY_CONFIG = {
|
|
deconz.const.CONF_ALLOW_CLIP_SENSOR: True,
|
|
deconz.const.CONF_ALLOW_DECONZ_GROUPS: True,
|
|
deconz.config_flow.CONF_API_KEY: "ABCDEF",
|
|
deconz.config_flow.CONF_BRIDGEID: "0123456789",
|
|
deconz.config_flow.CONF_HOST: "1.2.3.4",
|
|
deconz.config_flow.CONF_PORT: 80
|
|
}
|
|
|
|
|
|
async def setup_gateway(hass, data, allow_clip_sensor=True):
|
|
"""Load the deCONZ sensor platform."""
|
|
from pydeconz import DeconzSession
|
|
|
|
response = Mock(
|
|
status=200, json=asynctest.CoroutineMock(),
|
|
text=asynctest.CoroutineMock())
|
|
response.content_type = 'application/json'
|
|
|
|
session = Mock(
|
|
put=asynctest.CoroutineMock(
|
|
return_value=response
|
|
)
|
|
)
|
|
|
|
ENTRY_CONFIG[deconz.const.CONF_ALLOW_CLIP_SENSOR] = allow_clip_sensor
|
|
|
|
config_entry = config_entries.ConfigEntry(
|
|
1, deconz.DOMAIN, 'Mock Title', ENTRY_CONFIG, 'test',
|
|
config_entries.CONN_CLASS_LOCAL_PUSH)
|
|
gateway = deconz.DeconzGateway(hass, config_entry)
|
|
gateway.api = DeconzSession(hass.loop, session, **config_entry.data)
|
|
gateway.api.config = Mock()
|
|
hass.data[deconz.DOMAIN] = {gateway.bridgeid: gateway}
|
|
|
|
with patch('pydeconz.DeconzSession.async_get_state',
|
|
return_value=mock_coro(data)):
|
|
await gateway.api.async_load_parameters()
|
|
|
|
await hass.config_entries.async_forward_entry_setup(
|
|
config_entry, 'climate')
|
|
# To flush out the service call to update the group
|
|
await hass.async_block_till_done()
|
|
return gateway
|
|
|
|
|
|
async def test_platform_manually_configured(hass):
|
|
"""Test that we do not discover anything or try to set up a gateway."""
|
|
assert await async_setup_component(hass, climate.DOMAIN, {
|
|
'climate': {
|
|
'platform': deconz.DOMAIN
|
|
}
|
|
}) is True
|
|
assert deconz.DOMAIN not in hass.data
|
|
|
|
|
|
async def test_no_sensors(hass):
|
|
"""Test that no sensors in deconz results in no climate entities."""
|
|
gateway = await setup_gateway(hass, {})
|
|
assert not hass.data[deconz.DOMAIN][gateway.bridgeid].deconz_ids
|
|
assert not hass.states.async_all()
|
|
|
|
|
|
async def test_climate_devices(hass):
|
|
"""Test successful creation of sensor entities."""
|
|
gateway = await setup_gateway(hass, {"sensors": deepcopy(SENSOR)})
|
|
assert "climate.climate_1_name" in gateway.deconz_ids
|
|
assert "sensor.sensor_2_name" not in gateway.deconz_ids
|
|
assert len(hass.states.async_all()) == 1
|
|
|
|
gateway.api.sensors['1'].async_update(
|
|
{'state': {'on': False}})
|
|
|
|
await hass.services.async_call(
|
|
'climate', 'set_hvac_mode',
|
|
{'entity_id': 'climate.climate_1_name', 'hvac_mode': 'heat'},
|
|
blocking=True
|
|
)
|
|
gateway.api.session.put.assert_called_with(
|
|
'http://1.2.3.4:80/api/ABCDEF/sensors/1/config',
|
|
data='{"mode": "auto"}'
|
|
)
|
|
|
|
await hass.services.async_call(
|
|
'climate', 'set_hvac_mode',
|
|
{'entity_id': 'climate.climate_1_name', 'hvac_mode': 'off'},
|
|
blocking=True
|
|
)
|
|
gateway.api.session.put.assert_called_with(
|
|
'http://1.2.3.4:80/api/ABCDEF/sensors/1/config',
|
|
data='{"mode": "off"}'
|
|
)
|
|
|
|
await hass.services.async_call(
|
|
'climate', 'set_temperature',
|
|
{'entity_id': 'climate.climate_1_name', 'temperature': 20},
|
|
blocking=True
|
|
)
|
|
gateway.api.session.put.assert_called_with(
|
|
'http://1.2.3.4:80/api/ABCDEF/sensors/1/config',
|
|
data='{"heatsetpoint": 2000.0}'
|
|
)
|
|
|
|
assert len(gateway.api.session.put.mock_calls) == 3
|
|
|
|
|
|
async def test_verify_state_update(hass):
|
|
"""Test that state update properly."""
|
|
gateway = await setup_gateway(hass, {"sensors": deepcopy(SENSOR)})
|
|
assert "climate.climate_1_name" in gateway.deconz_ids
|
|
|
|
thermostat = hass.states.get('climate.climate_1_name')
|
|
assert thermostat.state == 'off'
|
|
|
|
state_update = {
|
|
"t": "event",
|
|
"e": "changed",
|
|
"r": "sensors",
|
|
"id": "1",
|
|
"state": {"on": False}
|
|
}
|
|
gateway.api.async_event_handler(state_update)
|
|
|
|
await hass.async_block_till_done()
|
|
assert len(hass.states.async_all()) == 1
|
|
|
|
thermostat = hass.states.get('climate.climate_1_name')
|
|
assert thermostat.state == 'off'
|
|
assert gateway.api.sensors['1'].changed_keys == \
|
|
{'state', 'r', 't', 'on', 'e', 'id'}
|
|
|
|
|
|
async def test_add_new_climate_device(hass):
|
|
"""Test successful creation of climate entities."""
|
|
gateway = await setup_gateway(hass, {})
|
|
sensor = Mock()
|
|
sensor.name = 'name'
|
|
sensor.type = 'ZHAThermostat'
|
|
sensor.register_async_callback = Mock()
|
|
async_dispatcher_send(
|
|
hass, gateway.async_event_new_device('sensor'), [sensor])
|
|
await hass.async_block_till_done()
|
|
assert "climate.name" in gateway.deconz_ids
|
|
|
|
|
|
async def test_do_not_allow_clipsensor(hass):
|
|
"""Test that clip sensors can be ignored."""
|
|
gateway = await setup_gateway(hass, {}, allow_clip_sensor=False)
|
|
sensor = Mock()
|
|
sensor.name = 'name'
|
|
sensor.type = 'CLIPThermostat'
|
|
sensor.register_async_callback = Mock()
|
|
async_dispatcher_send(
|
|
hass, gateway.async_event_new_device('sensor'), [sensor])
|
|
await hass.async_block_till_done()
|
|
assert len(gateway.deconz_ids) == 0
|
|
|
|
|
|
async def test_unload_sensor(hass):
|
|
"""Test that it works to unload sensor entities."""
|
|
gateway = await setup_gateway(hass, {"sensors": SENSOR})
|
|
|
|
await gateway.async_reset()
|
|
|
|
assert len(hass.states.async_all()) == 0
|