Skip to content

Commit 0b208f2

Browse files
committed
The webserver now looks ad data directly from mySQL
Added alert signal for data not seen for a long day (time turns red) Added a serialmonitor / comunicator page
1 parent 52d18bd commit 0b208f2

File tree

5 files changed

+284
-21
lines changed

5 files changed

+284
-21
lines changed

smart_incubator/server/smart_controller.py

+83-14
Original file line numberDiff line numberDiff line change
@@ -116,6 +116,23 @@ def retrieve_day(self, incubator, days=0, full=False):
116116
data = self.query(select_query)
117117
return data
118118

119+
def getAllIDs(self):
120+
"""
121+
"""
122+
select_query = "SELECT DISTINCT(id) as id FROM incubators ORDER BY id ASC;"
123+
data = self.query(select_query)
124+
return [i['id'] for i in data]
125+
126+
def retrieve_last_line(self, incubator):
127+
"""
128+
"""
129+
if incubator == 'all' or incubator < 0:
130+
select_query = "SELECT * FROM incubators.incubators WHERE device_time IN (SELECT MAX(device_time) FROM incubators.incubators GROUP BY id) ORDER BY id;"
131+
else:
132+
select_query = "SELECT * FROM incubators WHERE id = %s ORDER BY device_time DESC LIMIT 1;" % incubator
133+
134+
data = self.query(select_query)
135+
return data
119136

120137
class SerialController(threading.Thread):
121138
def __init__(self, port=None, baud=115200, filename=None, db_credentials=None):
@@ -132,7 +149,9 @@ def __init__(self, port=None, baud=115200, filename=None, db_credentials=None):
132149
self._delta_time_threshold = 15 # we sync time when and only when device had drifter more than this value (seconds)
133150
self._dd_mode_map = {0:'DD', 1:'LD', 2:'LL', 3:'DL', 4:'MM'}
134151

135-
self._datatail = collections.deque(maxlen=100) #keeps last 100 lines in memory
152+
self._datatail = collections.deque(maxlen=100) #keeps last 100 lines in memory as dict
153+
self._serial_queue = collections.deque(maxlen=100) #keeps last 100 lines in memory as raw lines
154+
136155
self._is_stopped = False
137156

138157
self._filename = filename
@@ -265,8 +284,21 @@ def getHistory(self, incubator,days=0):
265284
if self._database:
266285
return self._database.retrieve_day(incubator, days)
267286

268-
269287
def getlastData(self, incubator, json_mode=True):
288+
"""
289+
"""
290+
if self._database:
291+
if json_mode:
292+
return json.dumps(self._database.retrieve_last_line(incubator))
293+
else:
294+
return self._database.retrieve_last_line(incubator)
295+
296+
def getSerialBuffer(self):
297+
"""
298+
"""
299+
return self._serial_queue
300+
301+
def getlastDataFromBuffer(self, incubator, json_mode=True):
270302
"""
271303
"""
272304
if incubator == "all" or incubator < 0 :
@@ -301,6 +333,15 @@ def __iter__(self):
301333
self._sync_time( fields["id"], fields["device_time"])
302334
yield fields
303335

336+
def sendRaw(self, line):
337+
"""
338+
"""
339+
line = line.strip().encode('ascii')
340+
logging.debug("Sending " + line)
341+
self._serial.write(line + '\n')
342+
return True
343+
344+
304345
def sendCommand(self, inc_id, cmd, value):
305346
"""
306347
"""
@@ -365,15 +406,21 @@ def run(self):
365406
"""
366407
while not self._is_stopped:
367408
serial_line = self._serial.readline()
409+
self._serial_queue.append(serial_line)
410+
368411
fields = self._parse_serial_line(serial_line)
369412
if fields is None:
370413
continue
371414

372415
self._sync_time( fields["id"], fields["device_time"])
373-
self._datatail.append(fields) # a queue is kept for frequent access of recent events
416+
417+
418+
if self._filename:
419+
self._write_row_to_file(fields)
420+
self._datatail.append(fields)
374421

375-
if self._filename: self._write_row_to_file(fields)
376-
elif self._database: self._database.insert_row(fields)
422+
elif self._database:
423+
self._database.insert_row(fields)
377424

378425

379426

@@ -392,22 +439,35 @@ def __init__(self, host, port, serial_fetcher):
392439
self._serial_fetcher.start()
393440

394441
def _route(self):
395-
self._app.route('/', method="GET", callback=self._index)
396-
self._app.route('/static/<filepath:path>', callback=self._get_static)
397-
self._app.route('/json/<inc_id>', callback=self._incubator_json)
398-
self._app.route('/incubator/<inc_id>/<days>', callback=self._get_incubator)
442+
self._app.get('/', callback=self._index)
443+
self._app.get('/serial', callback=self._serialmonitor)
399444
self._app.route('/graph/<inc_id>', callback=self._get_graph, method=["post", "get"])
400-
self._app.route('/listen/<status>', callback=self._listen_to_serial)
401-
self._app.route('/quit', callback=self._quit)
445+
446+
self._app.get('/json/<inc_id>', callback=self._incubator_json)
447+
self._app.get('/incubator/<inc_id>/<days>', callback=self._get_incubator)
448+
449+
self._app.post('/send', callback=self._send_to_serial)
450+
451+
self._app.get('/static/<filepath:path>', callback=self._get_static)
452+
self._app.get('/listen/<status>', callback=self._listen_to_serial)
453+
self._app.get('/quit', callback=self._quit)
402454

403455
def start(self):
404456
self._app.run(host=self._host, port=self._port)
405457

458+
def _get_static(self, filepath):
459+
return static_file(filepath, root="./static")
460+
406461
def _index(self):
407462
return static_file('index.html', root="static")
463+
464+
def _serialmonitor(self):
465+
return static_file('serialmonitor.html', root="static")
408466

409-
def _get_static(self, filepath):
410-
return static_file(filepath, root="./static")
467+
def _send_to_serial(self):
468+
myDict = request.json['myDict']
469+
self._serial_fetcher.sendRaw ( myDict['line'] )
470+
return {"result": "OK"}
411471

412472
def _get_graph(self, inc_id):
413473

@@ -428,7 +488,16 @@ def _get_graph(self, inc_id):
428488
return template('static/graph.tpl', rep)
429489

430490
def _incubator_json(self, inc_id=0):
431-
return self._serial_fetcher.getlastData(inc_id)
491+
"""
492+
"""
493+
if inc_id == 'serial':
494+
495+
serial = list(self._serial_fetcher.getSerialBuffer())
496+
serial.reverse()
497+
data = {'result': ''.join(serial) }
498+
return data
499+
else:
500+
return self._serial_fetcher.getlastData(inc_id)
432501

433502
def _get_incubator(self, inc_id, days):
434503

smart_incubator/server/static/css/style.css

+35
Original file line numberDiff line numberDiff line change
@@ -55,6 +55,18 @@
5555
color: #777;
5656
text-decoration: none;
5757
}
58+
59+
.current-time {
60+
color: #777;
61+
float: left;
62+
padding: 15px 15px;
63+
font-size: 14px;
64+
line-height: 20px;
65+
font-family: "Helvetica Neue",Helvetica,Arial,sans-serif;
66+
color: #777;
67+
text-decoration: none;
68+
}
69+
5870

5971
.navbar-right {
6072
float: right!important;
@@ -336,3 +348,26 @@
336348
margin-right: 0px;
337349
float: right;
338350
}
351+
352+
.serialmonitor {
353+
margin:0 auto;
354+
overflow:hidden;
355+
float:left;
356+
align: center;
357+
#display: block;
358+
width: 1170px;
359+
#padding-left: 200px;
360+
#padding-right: 200px;
361+
362+
}
363+
364+
.serialmonitor #serialoutput {
365+
width: 80%;
366+
height: 250px;
367+
}
368+
369+
.serialmonitor #serialinput {
370+
width: 80%;
371+
height: 50px;
372+
}
373+

smart_incubator/server/static/index.html

+8-3
Original file line numberDiff line numberDiff line change
@@ -52,6 +52,7 @@
5252
$(document).ready(function(){
5353
loadDashboard();
5454
refreshDashboard();
55+
checkTimeValues();
5556
});
5657
</script>
5758

@@ -64,16 +65,20 @@
6465
<div class="container">
6566
<div class="navbar-header">
6667
<a class="navbar-brand" href="/">Incubators</a>
68+
<div class="current-time"></div>
6769
</div>
68-
6970
<ul class="nav navbar-nav navbar-right">
70-
<li><a href="localhost/" target="_blank">Ethoscopes</a></li>
71+
<li><a href="/serial" target="_blank">SerialMonitor </a></li>
72+
<li><a href="http://etho-node.lab.gilest.ro" target="_blank">Ethoscopes</a></li>
7173
</ul>
7274
</div>
7375
</nav>
7476
</header>
7577

76-
<div class="dashboard container" id="main"></div>
78+
<div class="dashboard container" id="main">
79+
<div class="serialmonitor"></div>
80+
</div>
81+
7782
</body>
7883

7984
</html>

smart_incubator/server/static/js/script.js

+52-4
Original file line numberDiff line numberDiff line change
@@ -51,8 +51,17 @@ function refreshDashboard(){
5151
function(data) {
5252
$.each(data, function(i, item) {
5353

54-
var time = moment(item.device_time*1000).format("DD-MM-YYYY HH:mm");
55-
var temperature = item.temperature;
54+
is_timestamp = new Date(item.device_time*1000).getTime() > 0;
55+
56+
if (is_timestamp) {
57+
var timestamp = item.device_time;
58+
} else {
59+
var k = moment(item.device_time);
60+
var timestamp = (k + k.utcOffset()*60*1000)/1000;
61+
}
62+
63+
var time = moment(timestamp*1000).format("DD-MM-YYYY HH:mm");
64+
var temperature = item.temperature;
5665

5766
if (Math.abs(item.temperature - item.set_temp ) > 0.5)
5867
{ $('#' + item.id).find('.temperature').css('color','#934c4c'); }
@@ -72,13 +81,52 @@ function refreshDashboard(){
7281
$('#' + item.id).find('.humidity').html(item.humidity);
7382
$('#' + item.id).find('.humidity').attr('title', item.humidity + " / " + item.set_hum);
7483

75-
$('#' + item.id).find('.time').find("p").html(time);
84+
$('#' + item.id).find('.time').find("p").html(item.device_time);
85+
$('#' + item.id).find('.time').attr('title', timestamp);
7686

7787
})
7888
});
79-
window.setTimeout(refreshDashboard,5000);
89+
window.setTimeout(refreshDashboard,10000);
8090
}
8191

92+
function refreshSerialMonitor(){
93+
94+
updateClock();
95+
$.getJSON(
96+
"/json/serial",
97+
function (data) {
98+
$('#serialoutput').val(data.result);
99+
}
100+
);
101+
window.setTimeout(refreshSerialMonitor,1000);
102+
}
103+
104+
function updateClock() {
105+
$('.current-time').text(moment().utc());
106+
}
107+
108+
function checkTimeValues(){
109+
110+
111+
var now = moment().utc();
112+
updateClock();
113+
114+
$('.time').each( function (i) {
115+
116+
var timestamp = $(this).attr('title');
117+
var time = moment( timestamp * 1000);
118+
var delta = now - time; // in ms
119+
120+
alert_range = 15 * 60 * 1000; //15minutes
121+
122+
if (delta >= alert_range) {
123+
$(this).find('p').css('color','#934c4c');
124+
} else {
125+
$(this).find('p').css('color','#777');
126+
}
127+
})
128+
window.setTimeout(checkTimeValues,1000)
129+
}
82130

83131
function show_alert_box() {
84132

0 commit comments

Comments
 (0)