]> jspc29.x-matter.uni-frankfurt.de Git - labtools.git/commitdiff
opus20: auto update for the status page / more graphs
authorPhilipp Klaus <klaus@physik.uni-frankfurt.de>
Wed, 1 Jul 2015 10:01:41 +0000 (12:01 +0200)
committerPhilipp Klaus <klaus@physik.uni-frankfurt.de>
Wed, 1 Jul 2015 10:01:41 +0000 (12:01 +0200)
opus20/opus20/opus20.py
opus20/opus20/webapp/__init__.py
opus20/opus20/webapp/static/css/style.css
opus20/opus20/webapp/static/js/script.js
opus20/opus20/webapp/views/base.jinja2
opus20/opus20/webapp/views/plots.jinja2
opus20/opus20/webapp/views/status.jinja2

index 482c6926f5dafbad50133d3a0cf66eaea53dd76d..6be8409639fe72cc94f74d70fe659f9985cbcee6 100644 (file)
@@ -691,6 +691,9 @@ class Object(object):
     def __str__(self):
         return repr(self)
 
+    def to_dict(self):
+        return self.__dict__.copy()
+
     @classmethod
     def from_dict(cls, d):
         o = Object()
index 2470e20e8e9e6e7e0af06c98b18569aef3f256bc..4a307dede5ea3bfc5cc630962cb0a68f37d615b5 100755 (executable)
@@ -50,9 +50,9 @@ class PlotWebServer(Bottle):
         self._current_values_last_call = -1E13
         self._download_device_data_last_call = -1E13
         super(PlotWebServer, self).__init__()
-        self.route('/connected/device', callback = self._connected_device)
         self.route('/list/devices', callback = self._list_devices)
         self.route('/download/<device_id>', callback = self._download_device_data)
+        self.route('/status/<device_id>', callback = self._status_device)
         self.route('/plot/<device_id>_history.<fileformat>', callback = self._plot_history)
         self.route('/static/<filename:path>', callback = self._serve_static)
         if self.debug: self.route('/debug', callback = self._debug_page)
@@ -61,13 +61,14 @@ class PlotWebServer(Bottle):
         self.route('/', callback = self._status_page)
 
     def _atg(self, vals):
-        """ Add template globals """
+        """ Add template globals
+            A wrapper function for the templated routes decorated with a @view() """
         vals.update(self.TPL_GLOBALS)
         return vals
 
     @view('status.jinja2')
     def _status_page(self):
-        return self._atg({'current_values': self.current_values, 'active': 'status'})
+        return self._atg({'device_id': self._connected_device, 'current_values': self.current_values, 'active': 'status'})
 
     @view('about.jinja2')
     def _about_page(self):
@@ -75,7 +76,7 @@ class PlotWebServer(Bottle):
 
     @view('plots.jinja2')
     def _plots_page(self):
-        return self._atg({'device_id': self._connected_device(), 'active': 'plots'})
+        return self._atg({'device_id': self._connected_device, 'active': 'plots'})
 
     @view('debug.jinja2')
     def _debug_page(self):
@@ -105,14 +106,29 @@ class PlotWebServer(Bottle):
         self._cached_current_values = cur
         return cur
 
+    def _status_device(self, device_id):
+        assert device_id == self._connected_device
+        status = self.current_values.to_dict()
+        status['ts'] = status['ts'].isoformat()
+        return {
+                'success': True,
+                'status': status,
+               }
+
     def _serve_static(self, filename):
         return static_file(filename, root=os.path.join(PATH, 'static'))
 
+    @property
     def _connected_device(self):
+        """ As long as the webserver can handle only a single
+            OPUS20 device, we return this single one here. """
         return self.o20.device_id
 
     def _list_devices(self):
-        return dict(devices=self.ps.get_device_ids())
+        return {
+                'success': True,
+                'devices': list(set([self._connected_device]) | set(self.ps.get_device_ids()))
+               }
 
     def _download_device_data(self, device_id):
         if  clock() - self._download_device_data_last_call < 10.0:
@@ -170,18 +186,16 @@ class PlotWebServer(Bottle):
         else:
             right = right.split(',')
         #selected_cols = df.columns
-        selected_cols = set()
+        selected_cols = []
         for measure in measures:
             for col in df.columns:
                 if measure in col:
-                    selected_cols.add(col)
-        selected_cols = list(selected_cols)
-        right_cols = set()
+                    if col not in selected_cols: selected_cols.append(col)
+        right_cols = []
         for col in selected_cols:
             for measure in right:
                 if measure in col:
-                    right_cols.add(col)
-        right_cols = list(right_cols)
+                    if col not in right_cols: right_cols.append(col)
         # / End handling URL query variables
 
         fig, ax = plt.subplots(figsize=figsize)
index de1218d0042b019e296dab4aa59788320475c658..37786d3511df0799be7222f24a7c6bd4f05980f3 100644 (file)
@@ -13,3 +13,7 @@ body {
 pre {
   text-align: left;
 }
+
+.chart {
+  margin: 10px 20px;
+}
index dad1e8a7ed0667dbb1164e8f66c5dca929803513..8b137891791fe96927ad78e64b0aad7bded08bdc 100644 (file)
@@ -1,21 +1 @@
 
-
-$(function(){
-  $('#fetchData').on('click', function(){
-    var $btn = $(this).button('loading')
-    $.ajax({
-      url: "/download/" + device_id,
-      type: 'get',
-      success: function (response) {
-        //console.log('response received');
-        location.reload()
-        $btn.button('reset')
-      }, error: function (response) {
-        console.log('ajax request to fetch data failed');
-        alert('ajax request to fetch data failed');
-        $btn.button('reset')
-      },
-    });
-  });
-});
-
index b75fade03366b319e93724c1d3715b79a2af1445..959195a9d8ccb48518eb2614f93794c2b20d2b16 100644 (file)
@@ -21,7 +21,7 @@
     <link href="/static/css/style.css" rel="stylesheet">
 
     <script type="text/javascript">
-          {% block globals %}{% endblock %}
+          {% block top_javascript %}{% endblock %}
     </script>
   </head>
 
     <script src="https://ajax.googleapis.com/ajax/libs/jquery/1.11.3/jquery.min.js"></script>
     <script src="https://maxcdn.bootstrapcdn.com/bootstrap/3.3.5/js/bootstrap.min.js"></script>
     <script src="/static/js/script.js"></script>
+
+    <script type="text/javascript">
+          {% block bottom_javascript %}{% endblock %}
+    </script>
     <!-- Placed at the end of the document so the pages load faster -->
   </body>
 </html>
index c77bb1c7371985b802bc02be4bf37f9c9cd919e3..22a92949af1c79a082cafc881643d09046ad1c51 100644 (file)
@@ -2,20 +2,56 @@
 
 {% block title %}Plot{% endblock %}
 
-{% block globals %}
+{% block top_javascript %}
+
 var device_id = "{{ device_id }}";
+
+$(function(){
+  $('#fetchData').on('click', function(){
+    var $btn = $(this);
+    $btn.prop('disabled', true);
+    var $text = $btn[0].textContent;
+    $btn.prop('textContent', "Loading...");
+    $.ajax({
+      url: "/download/" + device_id,
+      type: 'get',
+      success: function (response) {
+        //console.log('response received');
+        $btn.prop('disabled', false);
+        $btn.prop('textContent', $text);
+        location.reload()
+      }, error: function (response) {
+        console.log('ajax request to fetch data failed');
+        alert('ajax request to fetch data failed');
+        $btn.prop('disabled', false);
+        $btn.prop('textContent', $text);
+      },
+    });
+  });
+});
+
+
 {% endblock %}
 
 {% block content %}
             <h1 class="cover-heading">Plots of the Environment Values</h1> <br />
+            <h3>Combined Plot</h3>
+            <p class="lead">
+              <img class="chart" src="/plot/{{ device_id }}_history.png?figsize=10,7&measures=temperature,relative humidity&right=humidity" />
+            </p>
+            <h3>Temperature Plot</h3>
+            <p class="lead">
+              <img class="chart" src="/plot/{{ device_id }}_history.png?figsize=10,7&measures=temperature&color=blue" />
+            </p>
+            <h3>Relative Humidity Plot</h3>
             <p class="lead">
-              <!-- <img src="/plot/{{ device_id }}_history.png?figsize=10,7&measures=temperature,relative humidity&right=humidity" /> -->
-              <img src="/plot/{{ device_id }}_history.png?figsize=10,7&measures=temperature&color=blue" />
+              <img class="chart" src="/plot/{{ device_id }}_history.png?figsize=10,7&measures=relative humidity&right=-&ylabel=humidity [%25]&color=magenta" />
             </p>
+            <h3>Absolute Humidity Plot</h3>
             <p class="lead">
-              <img src="/plot/{{ device_id }}_history.png?figsize=10,7&measures=relative humidity&right=-&ylabel=humidity [%25]&color=magenta" />
+              <img class="chart" src="/plot/{{ device_id }}_history.png?figsize=10,7&measures=absolute humidity&right=-&ylabel=humidity [g%2Fm%B3]&color=cyan" />
             </p>
             <p class="lead">
-              <button id="fetchData" class="btn btn-lg btn-default" type="button" data-loading-text="Fetching New Data..." >Fetch New Data</button>
+              <button id="fetchData" class="btn btn-lg btn-default" type="button">Fetch New Data</button>
             </p>
 {% endblock %}
index a892402b54f3916e9d8f7136fec923458af03513..4128726395c4aa6266fe6c203ad476fa8aac566a 100644 (file)
@@ -2,17 +2,55 @@
 
 {% block title %}Status{% endblock %}
 
+{% block top_javascript %}
+
+var update_every_ms = 2050;
+var device_id = "{{ device_id }}";
+
+{% endblock %}
+
+{% block bottom_javascript %}
+
+function updateStatus() {
+  $.getJSON("/status/" + device_id, function( data ) {
+    $("#device_id").text(data.status.device_id);
+    $("#temperature").text(data.status.temperature.toFixed(1));
+    $("#dewpoint").text(data.status.dewpoint.toFixed(1));
+    $("#relative_humidity").text(data.status.relative_humidity.toFixed(1));
+    $("#absolute_humidity").text(data.status.absolute_humidity.toFixed(1));
+    $("#battery_voltage").text(data.status.battery_voltage.toFixed(1));
+    $("#timestamp").text(data.status.ts.replace('T', ' - '));
+  });
+};
+
+function timedUpdate() {
+  updateStatus();
+  setTimeout(function() {
+    timedUpdate();
+  }, update_every_ms);
+};
+
+$(function() {
+  // on page load
+  updateStatus();
+  setTimeout(function() {
+    timedUpdate();
+  }, update_every_ms);
+});
+
+{% endblock %}
+
 {% block content %}
             <h1 class="cover-heading">Current Status</h1>
             <p class="lead">
-              <div>Device ID:         {{ current_values.device_id                 }}</div>
-              <div>Temperature:       {{ current_values.temperature       | round(1) }} °C</div>
-              <div>Dewpoint:          {{ current_values.dewpoint          | round(1) }} °C</div>
-              <div>Relative Humidity: {{ current_values.relative_humidity | round(1) }} %</div>
-              <div>Absolute Humidity: {{ current_values.absolute_humidity | round(1) }} g/m³</div>
-              <div>Battery Voltage:   {{ current_values.battery_voltage   | round(1) }} V</div>
+              <div>Device ID:                 <span id="device_id">{{ current_values.device_id                    }}</span>      </div>
+              <div>Temperature:             <span id="temperature">{{ current_values.temperature       | round(1) }}</span> °C   </div>
+              <div>Dewpoint:                   <span id="dewpoint">{{ current_values.dewpoint          | round(1) }}</span> °C   </div>
+              <div>Relative Humidity: <span id="relative_humidity">{{ current_values.relative_humidity | round(1) }}</span> %    </div>
+              <div>Absolute Humidity: <span id="absolute_humidity">{{ current_values.absolute_humidity | round(1) }}</span> g/m³ </div>
+              <div>Battery Voltage:     <span id="battery_voltage">{{ current_values.battery_voltage   | round(1) }}</span> V    </div>
               <br />
-              <div>Date &amp; Time of this information: {{ current_values.ts.isoformat().replace('T', ' - ') }}</div>
+              <div>Date &amp; Time of this information: <span id="timestamp">{{ current_values.ts.isoformat().replace('T', ' - ') }}</span></div>
             </p>
             <!--<p class="lead">
               <a href="#" class="btn btn-lg btn-default">Learn more</a>