Skip to content

Commit b2ad9ca

Browse files
committed
Added support for browsing graphs by day
Hopefully fixed issues with db connection
1 parent f24f7d2 commit b2ad9ca

File tree

4 files changed

+73
-32
lines changed

4 files changed

+73
-32
lines changed

smart_incubator/server/smart_controller.py

+33-28
Original file line numberDiff line numberDiff line change
@@ -13,6 +13,7 @@
1313
import json
1414

1515
import MySQLdb
16+
import MySQLdb.cursors
1617
from MySQLdb.constants import FIELD_TYPE
1718

1819

@@ -55,38 +56,42 @@ def connect(self):
5556
_db_user_name = self._db_credentials["username"]
5657
_db_user_pass = self._db_credentials["password"]
5758

58-
my_conv = { FIELD_TYPE.TIMESTAMP: str }
59-
60-
self.connection = MySQLdb.connect('127.0.0.1', _db_user_name, _db_user_pass, _db_name, conv=my_conv)
61-
self.cursor = self.connection.cursor(MySQLdb.cursors.DictCursor)
62-
self.cursor.connection.autocommit(True)
59+
my_conv = { FIELD_TYPE.TIMESTAMP: str, FIELD_TYPE.FLOAT: float, FIELD_TYPE.TINY: int, FIELD_TYPE.LONG: int, FIELD_TYPE.INT24: int }
60+
61+
logging.debug("Creating a new connection with the `incubators` db")
62+
self.connection = MySQLdb.connect('localhost', _db_user_name, _db_user_pass, _db_name, conv=my_conv, cursorclass=MySQLdb.cursors.DictCursor)
6363

6464
@staticmethod
6565
def _timestamp_to_datetime(timestamp):
6666
return time.strftime('%Y-%m-%d %H:%M:%S', time.gmtime(timestamp))
6767

6868
def insert(self, query):
6969
try:
70-
self.cursor.execute(query)
71-
self.connection.commit()
72-
self.connection.close()
73-
except:
70+
with self.connection as cursor:
71+
cursor.execute(query)
72+
self.connection.commit()
73+
74+
# For some reason, I often get a "OperationalError: (2006, 'MySQL server has gone away')"
75+
# Could not debug this - so, as workaround, I create a new connection if the existing one is broken
76+
# This, however, could result in a dangerous loop if connection keeps dropping
77+
except (AttributeError, MySQLdb.OperationalError):
7478

7579
self.connect()
76-
self.cursor.execute(query)
77-
self.connection.commit()
78-
self.connection.close()
80+
self.insert(query)
7981

8082
def query(self, query):
81-
self.connect()
82-
cursor = self.connection.cursor( MySQLdb.cursors.DictCursor )
83-
cursor.execute(query)
84-
result = cursor.fetchall()
85-
86-
#self.cursor.execute(query)
87-
#result = self.cursor.fetchall()
83+
try:
84+
85+
with self.connection as cursor:
86+
cursor.execute(query)
87+
result = cursor.fetchall()
8888

89-
self.connection.close()
89+
# For some reason, I often get a "OperationalError: (2006, 'MySQL server has gone away')"
90+
# Could not debug this - so, as workaround, I create a new connection if the existing one is broken
91+
except (AttributeError, MySQLdb.OperationalError):
92+
93+
self.connect()
94+
result = self.query(query)
9095

9196
return result
9297

@@ -149,7 +154,7 @@ def retrieve_last_line(self, incubator):
149154
else:
150155
select_query = "SELECT * FROM incubators WHERE id = %s ORDER BY device_time DESC LIMIT 1;" % incubator
151156
data = self.query(select_query)[0]
152-
157+
153158
return data
154159

155160
class SerialController(threading.Thread):
@@ -306,7 +311,7 @@ def getlastData(self, incubator, json_mode=True):
306311
"""
307312
"""
308313
if self._database:
309-
if json_mode or (incubator != 'all' and incubator >= 0 ):
314+
if json_mode or (incubator != 'all' and incubator < 0 ):
310315
return json.dumps(self._database.retrieve_last_line(incubator))
311316
else:
312317
return self._database.retrieve_last_line(incubator)
@@ -394,7 +399,7 @@ def sendCommand(self, inc_id, cmd, value):
394399
def update(self, inc_id, values):
395400
"""
396401
"""
397-
current = self.getlastData( inc_id )
402+
current = self.getlastData( inc_id , json_mode=False)
398403

399404
resp = ""
400405

@@ -404,7 +409,7 @@ def update(self, inc_id, values):
404409
if values['set_hum'] != current['set_hum'] :
405410
if self.sendCommand(inc_id, cmd='set_hum', value=values['set_hum']):
406411
resp += "Humidity set to %s\n" % values['set_hum']
407-
if values['set_light'] != current['set_light'] :
412+
if values['set_light'] != int(current['set_light']) : # why light is not being converted by mysqldb??!
408413
if self.sendCommand(inc_id, cmd='set_light', value=values['set_light']):
409414
resp += "Light set to %s\n" % values['set_light']
410415
if values['dd_mode'] != current['dd_mode'] :
@@ -459,7 +464,7 @@ def __init__(self, host, port, serial_fetcher):
459464
def _route(self):
460465
self._app.get('/', callback=self._index)
461466
self._app.get('/serial', callback=self._serialmonitor)
462-
self._app.route('/graph/<inc_id>', callback=self._get_graph, method=["post", "get"])
467+
self._app.route('/graph/<inc_id>/<day>', callback=self._get_graph, method=["post", "get"])
463468

464469
self._app.get('/json/<inc_id>', callback=self._incubator_json)
465470
self._app.get('/incubator/<inc_id>/<days>', callback=self._get_incubator)
@@ -487,7 +492,7 @@ def _send_to_serial(self):
487492
self._serial_fetcher.sendRaw ( myDict['line'] )
488493
return {"result": "OK"}
489494

490-
def _get_graph(self, inc_id):
495+
def _get_graph(self, inc_id, day):
491496

492497
rep = {}
493498
if request.forms.get("submitted"):
@@ -502,7 +507,8 @@ def _get_graph(self, inc_id):
502507
else:
503508
rep['message'] = ''
504509

505-
rep['incubator_id'] = inc_id
510+
rep['incubator_id'] = inc_id
511+
rep['day'] = day
506512
return template('static/graph.tpl', rep)
507513

508514
def _incubator_json(self, inc_id=0):
@@ -558,7 +564,6 @@ def transfer_file_to_db(filename, db_credentials):
558564
logging.debug("Logger in DEBUG mode")
559565

560566
db_credentials = {'username': 'incubators', 'password' : 'incubators', 'db_name' : 'incubators' }
561-
db = mySQLDatabase(db_credentials)
562567

563568
if option_dict['output']:
564569
serial_fetcher = SerialController(option_dict["port"], filename=option_dict['output'])

smart_incubator/server/static/css/style.css

+19
Original file line numberDiff line numberDiff line change
@@ -130,6 +130,25 @@
130130
text-align: left;
131131
box-shadow: 0 4px 8px 0 rgba(0, 0, 0, 0.2), 0 6px 20px 0 rgba(0, 0, 0, 0.19);
132132
}
133+
134+
#time-buttons{
135+
background: #ffffff;
136+
border-color: #e7e7e7;
137+
border: 2px;
138+
padding: 20px;
139+
float: left;
140+
margin-top: 50px;
141+
margin-left: 0px;
142+
margin-bottom: 0px;
143+
width: 100%;
144+
text-align: center;
145+
box-shadow: 0 4px 8px 0 rgba(0, 0, 0, 0.2), 0 6px 20px 0 rgba(0, 0, 0, 0.19);
146+
}
147+
148+
#time-buttons a {
149+
color: #777;
150+
text-decoration: none;
151+
}
133152

134153
form input[type=submit] {
135154
background-color: #777;

smart_incubator/server/static/graph.tpl

+8-2
Original file line numberDiff line numberDiff line change
@@ -49,9 +49,10 @@
4949
<script type="text/javascript">
5050
5151
$(document).ready(function(){
52-
loadGraphs({{incubator_id}}, 1);
52+
loadGraphs({{incubator_id}}, {{day}});
5353
loadIncubatorForm({{incubator_id}});
5454
show_alert_box();
55+
show_time_buttons({{incubator_id}}, {{day}});
5556
});
5657
5758
</script>
@@ -85,14 +86,19 @@
8586
<div class="graph lost">
8687
<div class="agile">
8788
<div class="w3l-agile">
88-
<h3>Incubator {{incubator_id}}</h3>
89+
<h3 id="graph-title">Incubator {{incubator_id}}</h3></h2>
8990
</div>
9091
<div id="temperature"></div>
9192
<div id="humidity"></div>
9293
<div id="light"></div>
9394
</div>
9495
</div>
9596
97+
<div class="agile">
98+
<div id="time-buttons"></div>
99+
</div>
100+
101+
96102
<div class="agile">
97103
<div id="incubator-data"></div>
98104
</div>

smart_incubator/server/static/js/script.js

+13-2
Original file line numberDiff line numberDiff line change
@@ -14,7 +14,7 @@ function loadDashboard(){
1414

1515
$('#main').append('\
1616
<div class="incubator" id="'+item.id+'">\
17-
<h2><a href="/graph/'+item.id+'">Incubator '+item.id+'</a></h2>\
17+
<h2><a href="/graph/'+item.id+'/1">Incubator '+item.id+'</a></h2>\
1818
<div class="data">\
1919
<p class="temperature">'+item.temperature+'</p><div class="light"></div></div>\
2020
<div class="data">\
@@ -103,7 +103,7 @@ function refreshDashboard(){
103103

104104
})
105105
});
106-
window.setTimeout(refreshDashboard,10000);
106+
window.setTimeout(refreshDashboard,1*60*1000); //we refresh once a minute
107107
}
108108

109109
function refreshSerialMonitor(){
@@ -154,7 +154,18 @@ function show_alert_box() {
154154

155155
}
156156

157+
function show_time_buttons(incubator_id, day) {
158+
159+
$("#time-buttons").html('<a id=\"previous-day\" href="/graph/'+incubator_id+'/'+(day+1)+'">Previous Day</a> - <a id=\"next-day\" href="/graph/'+incubator_id+'/'+(day-1)+'">Next Day</a>');
160+
161+
}
162+
157163
function loadGraphs(incubator_id, days){
164+
165+
dateFrom = moment().subtract(days,'d').format('YYYY MMM DD dddd');
166+
167+
$('#graph-title').text( $('#graph-title').text() + " - " + dateFrom )
168+
158169
$.ajax({
159170
url: "/incubator/"+incubator_id+"/"+days,
160171
dataType: 'JSON',

0 commit comments

Comments
 (0)