forked from martiby/ess
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathapi_request.py
111 lines (94 loc) · 3.69 KB
/
api_request.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
# 09.04.2022 Martin Steppuhn
# 15.08.2022 Martin Steppuhn error in answer dictionary
import json
import logging
import time
import requests
class ApiRequest:
"""
Simplified HTTP Request for JSON APIs.
Sends an HTTP request. The response (JSON) is converted to a dictionary.
With Lifetime a timeout for data can be specified. Even if there is an error or no read, data is still valid for
the specified period.
"""
def __init__(self, url, timeout=1, lifetime=10, log_name='api'):
"""
:param url: string
:param timeout: timeout for request in seconds
:param lifetime: timeout for data in seconds
:param log_name: name for logger
"""
self.url = url
self.timeout = timeout
self.lifetime = lifetime
self.log = logging.getLogger(log_name)
self.data = None # Data
self.lifetime_timeout = time.perf_counter() + self.lifetime if self.lifetime else None # set lifetime timeout
self.log.debug("init url: {}".format(url))
def read(self, post=None, url_extension=''):
"""
Read
:return: data (dictionary)
"""
t0 = time.perf_counter()
error = None
data = None
url = ""
try:
url = self.url + url_extension
if post is None:
r = requests.get(url, timeout=self.timeout)
else:
r = requests.post(url, timeout=self.timeout, json=post)
if r.status_code == 200:
data = json.loads(r.content)
else:
raise ValueError("status_code={} url={}".format(r.status_code, url))
self.lifetime_timeout = t0 + self.lifetime if self.lifetime else None # set new lifetime timeout
self.data = data # update data
self.log.debug("read done in {:.3f}s data: {}".format(time.perf_counter() - t0, data))
except requests.ConnectionError:
error = "connection to {} failed ({:.3f}s)".format(url, time.perf_counter() - t0)
self.log.error(error)
data = {'error': error}
except Exception as e:
error = "read to {} failed ({:.3f}s)".format(url, time.perf_counter() - t0)
self.log.error(error + " error: {}".format(e))
data = {'error': error}
if self.lifetime:
if self.lifetime_timeout and time.perf_counter() > self.lifetime_timeout:
self.lifetime_timeout = None # disable timeout, restart with next valid receive
self.data = data
else:
self.data = data # without lifetime set self.data instantly to read result
return data
def get(self, key, default=None):
"""
Get a value from data.
:param key: String (key for data dictionary, or list with nested keys and index)
:param default: return für invalid get
:return: value
"""
try:
value = self.data
if isinstance(key, (tuple, list)):
for k in key:
value = value[k]
else:
value = value[key]
return value
except:
return default
if __name__ == "__main__":
logging.basicConfig(
level=logging.DEBUG,
format='%(asctime)s %(name)-10s %(levelname)-6s %(message)s',
datefmt='%Y-%m-%d %H:%M:%S',
)
logging.getLogger("urllib3.connectionpool").setLevel(logging.INFO)
api = ApiRequest("http://192.168.0.10:8008", timeout=1, lifetime=10)
# api = ApiRequest("http://192.168.50.50/api/system/status", timeout=1, lifetime=10)
while True:
data = api.read()
print(data, api.data)
time.sleep(3)