-
Notifications
You must be signed in to change notification settings - Fork 14
/
Copy pathToon.py
131 lines (111 loc) · 5.82 KB
/
Toon.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
import re
import json
import sys
import uuid
import pprint
import requests
import time
""" The Toon module impersonates the Eneco Toon mobile app, and implements
similar functionality. """
class Toon:
""" Log in to the Toon API, and do stuff. """
def __init__(self, username, password):
self.username = username
self.password = password
self.toonstate = None
self.sessiondata = None
self.debug = 0
self.max_retries = 3
self.retry_interval = 1
self.required_datakeys = [ 'deviceStatusInfo', 'gasUsage', 'powerUsage', 'thermostatInfo' ]
def login(self):
""" Log in to the toon API, and set up a session. """
# First we open the login page, to get the agreement
# details.
# Agreement details and user details are stored in the
# sessiondata variable.
formdata = {"username": self.username,
"password": self.password}
r = requests.get("https://toonopafstand.eneco.nl/toonMobileBackendWeb/client/login",
params=formdata)
self.sessiondata = r.json()
def logout(self):
""" Log out of the API.
This is needed, as toon keeps a maximum amount of display clients.
Too many logins lead to 500 responses from the API. """
formdata = {"clientId": self.sessiondata["clientId"],
"clientIdChecksum": self.sessiondata["clientIdChecksum"],
"random": uuid.uuid4()}
r = requests.get("https://toonopafstand.eneco.nl/toonMobileBackendWeb/client/auth/logout", params=formdata)
self.toonstate = None
self.sessiondata = None
def set_maxretries(self,max_retries):
""" Set maximum of retries (default: 3). """
self.max_retries = max_retries
def retrieve_toon_state(self):
self.refresh_toon_state()
formdata = {"clientId": self.sessiondata["clientId"],
"clientIdChecksum": self.sessiondata["clientIdChecksum"],
"random": uuid.uuid4()}
self.toonstate = {}
retries = 0
while not all(x in self.toonstate for x in self.required_datakeys):
retries += 1
if retries > self.max_retries:
raise Exception('Incomplete response')
elif retries > 1:
time.sleep(self.retry_interval)
r = requests.get("https://toonopafstand.eneco.nl/toonMobileBackendWeb/client/auth/retrieveToonState", params=formdata)
if r.status_code == 200:
self.toonstate = r.json()
return
else:
self.logout()
raise Exception("retrieve status: ", r.status_code)
def refresh_toon_state(self):
# refreshing the session helps with the keep-alive.
# Now re-use the agreement details to do the actual
# authentication. This establishes a session, and allows
# state to be retrieved.
# TODO: check for variable existence / throw exception on
# failure
formdata = {"clientId": self.sessiondata["clientId"],
"clientIdChecksum": self.sessiondata["clientIdChecksum"],
"agreementId": self.sessiondata["agreements"][0]["agreementId"],
"agreementIdChecksum": self.sessiondata["agreements"][0]["agreementIdChecksum"],
"random": uuid.uuid4()}
r = requests.get("https://toonopafstand.eneco.nl/toonMobileBackendWeb/client/auth/start", params=formdata)
if r.status_code == 200:
return
else:
self.logout()
raise Exception("refresh status: ", r.status_code)
def get_gas_usage(self):
self.retrieve_toon_state()
return self.toonstate["gasUsage"]
def get_power_usage(self):
self.retrieve_toon_state()
return self.toonstate["powerUsage"]
def get_thermostat_info(self):
self.retrieve_toon_state()
return self.toonstate["thermostatInfo"]
def get_thermostat_states(self):
self.retrieve_toon_state()
return self.toonstate["thermostatStates"]
def set_thermostat(self, temperature):
targettemp = int(temperature*100)
formdata = {"clientId": self.sessiondata["clientId"],
"clientIdChecksum": self.sessiondata["clientIdChecksum"],
"value": targettemp,
"random": uuid.uuid1()}
r = requests.get("https://toonopafstand.eneco.nl/toonMobileBackendWeb/client/auth/setPoint", params=formdata)
def get_program_state(self):
self.retrieve_toon_state()
return self.toonstate["thermostatInfo"]["activeState"]
def set_program_state(self,targetstate):
formdata = {"clientId": self.sessiondata["clientId"],
"clientIdChecksum": self.sessiondata["clientIdChecksum"],
"state": 2,
"temperatureState": targetstate,
"random": uuid.uuid1()}
r = requests.get("https://toonopafstand.eneco.nl/toonMobileBackendWeb/client/auth/schemeState", params=formdata)