This repository has been archived by the owner on Mar 18, 2020. It is now read-only.
-
Notifications
You must be signed in to change notification settings - Fork 5
/
Copy pathe_thermostaat.py
295 lines (247 loc) · 10.3 KB
/
e_thermostaat.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
"""
Adds support for the Essent Icy E-Thermostaat units.
For more details about this platform, please refer to the documentation at
https://github.com/custom-components/climate.e_thermostaat
"""
import logging
import requests
import voluptuous as vol
from homeassistant.components.climate import ClimateDevice, PLATFORM_SCHEMA
try:
from homeassistant.components.climate.const import (
SUPPORT_OPERATION_MODE, SUPPORT_TARGET_TEMPERATURE, SUPPORT_AWAY_MODE)
except ImportError:
from homeassistant.components.climate import (
SUPPORT_OPERATION_MODE, SUPPORT_TARGET_TEMPERATURE, SUPPORT_AWAY_MODE)
from homeassistant.const import (
ATTR_TEMPERATURE, CONF_USERNAME, CONF_PASSWORD, TEMP_CELSIUS)
import homeassistant.helpers.config_validation as cv
__version__ = '0.2.1'
_LOGGER = logging.getLogger(__name__)
URL_LOGIN = "https://portal.icy.nl/login"
URL_DATA = "https://portal.icy.nl/data"
DEFAULT_NAME = 'E-Thermostaat'
CONF_NAME = 'name'
CONF_COMFORT_TEMPERATURE = 'comfort_temperature'
CONF_SAVING_TEMPERATURE = 'saving_temperature'
CONF_AWAY_TEMPERATURE = 'away_temperature'
STATE_COMFORT = "Comfortstand" # "comfort"
STATE_SAVING = "Bespaarstand" # "saving"
STATE_AWAY = "Lang-weg-stand" # "away"
STATE_FIXED_TEMP = "Vaste temperatuur" # "fixed temperature"
DEFAULT_COMFORT_TEMPERATURE = 20
DEFAULT_SAVING_TEMPERATURE = 17
DEFAULT_AWAY_TEMPERATURE = 12
# Values from web interface
MIN_TEMP = 10
MAX_TEMP = 30
# Values of E-Thermostaat to map to operation mode
COMFORT = 32
SAVING = 64
AWAY = 0
FIXED_TEMP = 128
SUPPORT_FLAGS = (SUPPORT_TARGET_TEMPERATURE | SUPPORT_OPERATION_MODE |
SUPPORT_AWAY_MODE)
PLATFORM_SCHEMA = PLATFORM_SCHEMA.extend({
vol.Optional(CONF_NAME, default=DEFAULT_NAME): cv.string,
vol.Required(CONF_USERNAME): cv.string,
vol.Required(CONF_PASSWORD): cv.string,
vol.Optional(CONF_AWAY_TEMPERATURE, default=DEFAULT_AWAY_TEMPERATURE):
vol.Coerce(float),
vol.Optional(CONF_SAVING_TEMPERATURE, default=DEFAULT_SAVING_TEMPERATURE):
vol.Coerce(float),
vol.Optional(CONF_COMFORT_TEMPERATURE,
default=DEFAULT_COMFORT_TEMPERATURE): vol.Coerce(float)
})
def setup_platform(hass, config, add_devices, discovery_info=None):
"""Setup the E-Thermostaat Platform."""
name = config.get(CONF_NAME)
comfort_temp = config.get(CONF_COMFORT_TEMPERATURE)
saving_temp = config.get(CONF_SAVING_TEMPERATURE)
away_temp = config.get(CONF_AWAY_TEMPERATURE)
username = config.get(CONF_USERNAME)
password = config.get(CONF_PASSWORD)
add_devices([EThermostaat(
name, username, password,
comfort_temp, saving_temp, away_temp)])
class EThermostaat(ClimateDevice):
"""Representation of a E-Thermostaat device."""
def __init__(self, name, username, password,
comfort_temp, saving_temp, away_temp):
"""Initialize the thermostat."""
self._name = name
self._username = username
self._password = password
self._comfort_temp = comfort_temp
self._saving_temp = saving_temp
self._away_temp = away_temp
self._current_temperature = None
self._target_temperature = None
self._old_conf = None
self._current_operation_mode = None
self._uid = None
self._token = None
self._operation_list = [STATE_COMFORT, STATE_SAVING,
STATE_AWAY, STATE_FIXED_TEMP]
self.update()
@property
def name(self):
"""Return the name of the Device."""
return self._name
@property
def should_poll(self):
"""Polling is required."""
return True
@property
def min_temp(self):
"""Return the minimum temperature."""
return MIN_TEMP
@property
def max_temp(self):
"""Return the maximum temperature."""
return MAX_TEMP
@property
def temperature_unit(self):
"""Return the unit of measurement."""
return TEMP_CELSIUS
@property
def current_temperature(self):
"""Return the current temperature."""
return self._current_temperature
@property
def target_temperature(self):
"""Return the temperature we try to reach."""
return self._target_temperature
@property
def operation_list(self):
"""List of available operation modes."""
return self._operation_list
@property
def current_operation(self):
"""Return current operation mode."""
return self._current_operation_mode
@property
def is_away_mode_on(self):
"""Return true if away mode is on."""
return self._current_operation_mode in [STATE_AWAY]
@property
def supported_features(self):
"""Return the list of supported features."""
return SUPPORT_FLAGS
def set_operation_mode(self, operation_mode):
"""Set new operation mode."""
if operation_mode == STATE_COMFORT:
self._set_temperature(self._comfort_temp, mode_int=COMFORT)
elif operation_mode == STATE_SAVING:
self._set_temperature(self._saving_temp, mode_int=SAVING)
elif operation_mode == STATE_AWAY:
self._set_temperature(self._away_temp, mode_int=AWAY)
elif operation_mode == STATE_FIXED_TEMP:
self._set_temperature(self._target_temperature,
mode_int=FIXED_TEMP)
def turn_away_mode_on(self):
"""Turn away on."""
self._set_temperature(self._away_temp, mode_int=AWAY)
def turn_away_mode_off(self):
"""Turn away off and set comfort temp."""
self._set_temperature(self._comfort_temp, mode_int=COMFORT)
def set_temperature(self, **kwargs):
"""Set new target temperature."""
temperature = kwargs.get(ATTR_TEMPERATURE)
if temperature is None:
return
self._set_temperature(temperature)
def _set_temperature(self, temperature, mode_int=None):
"""Set new target temperature, via URL commands."""
payload_new = [("temperature1", temperature)]
if self._old_conf is not None:
# overwrite first int to set mode
if mode_int is not None:
payload_new.append(("configuration[]", mode_int))
elif self._current_operation_mode \
and self._current_operation_mode == STATE_FIXED_TEMP:
payload_new.append(("configuration[]", FIXED_TEMP))
else:
payload_new.append(("configuration[]", COMFORT))
mode_int = payload_new[1][1]
# append old config not to overwrite old settings
for i in self._old_conf[1:]:
payload_new.append(('configuration[]', i))
if self._request_with_retry(URL_DATA, payload_new,
request_type='post'):
# update state values since we assume update was successful
self._current_operation_mode = \
self.map_int_to_operation_mode(mode_int)
self._target_temperature = temperature
# self.schedule_update_ha_state(force_refresh=True)
def get_token_and_uid(self):
"""Get the Session Token and UID of the Thermostaat."""
payload = {'username': self._username, 'password': self._password}
with requests.Session() as s:
try:
s.get(URL_LOGIN)
r = s.post(URL_LOGIN, data=payload)
res = r.json()
self._token = res['token']
self._uid = res['serialthermostat1']
except Exception as e:
_LOGGER.error("Could not get token and uid: %s" % e)
def _send_request_with_header(self, url, payload, request_type):
"""Sends a request with header set the uid if necessary."""
header = {'Session-token': self._token}
if request_type == 'get':
# for get data just use payload as is
payload_new = payload
req_func = requests.get
else:
# make a local copy to prevent overwriting
payload_new = [x for x in payload]
payload_new.append(("uid", self._uid))
req_func = requests.post
try:
r = req_func(url, data=payload_new, headers=header)
return r
except Exception as e:
_LOGGER.error("Could not connect to E-Thermostaat."
"error: %s" % e)
def _request_with_retry(self, url, payload, request_type):
"""Try once with old token, if fails get new token and try again."""
if self._uid is None or self._token is None:
self.get_token_and_uid()
r = self._send_request_with_header(url, payload, request_type)
# if unauthorized, get new token and uid
if r is None or r.json()['status']['code'] == 401:
self.get_token_and_uid()
r = self._send_request_with_header(url, payload, request_type)
# Code 200 is success
if r is None or not r.json()['status']['code'] == 200:
_LOGGER.error("Could not set temperature, "
"error on the server side")
return r
def _get_data(self):
"""Get the data of the E-Thermostaat."""
payload = (('username', self._username), ('password', self._password))
r = self._request_with_retry(URL_DATA, payload, request_type='get')
if r:
data = r.json()
self._target_temperature = data['temperature1']
self._current_temperature = data['temperature2']
self._old_conf = data['configuration']
self._current_operation_mode = \
self.map_int_to_operation_mode(self._old_conf[0])
_LOGGER.debug("E-Thermostaat value: {}".format(self._old_conf[0]))
else:
_LOGGER.error("Could not get data from E-Thermostaat.")
def update(self):
"""Get the latest data."""
self._get_data()
@staticmethod
def map_int_to_operation_mode(config_int):
"""Map the value of the E-Thermostaat to the operation mode."""
if AWAY <= config_int < COMFORT:
return STATE_AWAY
elif COMFORT <= config_int < SAVING:
return STATE_COMFORT
elif SAVING <= config_int < FIXED_TEMP:
return STATE_SAVING
return STATE_FIXED_TEMP