Skip to content
This repository has been archived by the owner on Nov 16, 2020. It is now read-only.

cloudless (standalone) freeboard dashboard for embedded system

Notifications You must be signed in to change notification settings

phyunsj/embedded-freeboard-dashboard

Repository files navigation

Freeboard Dashboard + Embedded WebApplication

  • A damn-sexy, open source real-time dashboard builder for IOT and other web mashups. A free open-source alternative to Geckoboard. from freeboard.io
  • Fast and easy to use C++ micro web framework from Crow
  • Flow-based programming for the Internet of Things from Node-RED

0. The Goal

  • Create a stand-alone (customized) version of Freeboard Dashboard
  • Customize HTTP Handler to respond polling requests from Freeboard Dashboard

1. Generate index.html (Development Version)

  • index-freeboard.html (edited from index.html) & Gruntfile.js
  • imgEmbed and processhtml target are added to build a single index.html with images + *.min.js + *.min.css
  • add additional plugins ( sparkline, gauge, ...)
plugins : {
                src : [
                    'plugins/freeboard/*.js',
+                   'plugins/thirdparty/*.js'
                ],
                dest : 'js/freeboard.plugins.js'
            },
  • additional dependencies

    npm install grunt-processhtml --save-dev

    npm install grunt-image-embed --save-dev

2. Create Datasource(s)

3. Create Pane(s)

4. Test Dashboard with Node-RED

Node-RED Flow

All data are simulated from Node-RED function node

[{"id":"1d9ece45.b4b6f2","type":"http in","z":"c77db134.66d82","name":"http://localhost:1880/freeboard","url":"/freeboard/index.html","method":"get","upload":false,"swaggerDoc":"","x":170.16664123535156,"y":120.66666793823242,"wires":[["84075ba0.7e8218"]]},{"id":"84075ba0.7e8218","type":"file in","z":"c77db134.66d82","name":"index.html","filename":"C:\\Users\\sphyun\\Documents\\project.freeboard\\dev\\index.html","format":"utf8","chunk":false,"sendError":false,"x":411.1666679382324,"y":121.55555629730225,"wires":[["b9badebf.2988f"]]},{"id":"b9badebf.2988f","type":"http response","z":"c77db134.66d82","name":"GET Response","statusCode":"","headers":{},"x":671.1666717529297,"y":202.4444456100464,"wires":[]},{"id":"f0427c5f.b54ad","type":"http in","z":"c77db134.66d82","name":"/freeboard/kitchen","url":"/freeboard/kitchen","method":"get","upload":false,"swaggerDoc":"","x":145.16667556762695,"y":216.88889074325562,"wires":[["f2fe075f.2e6928"]]},{"id":"cc286368.74c28","type":"debug","z":"c77db134.66d82","name":"","active":true,"tosidebar":true,"console":false,"tostatus":false,"complete":"false","x":663.1666717529297,"y":260.33333587646484,"wires":[]},{"id":"1d82e6e2.a9a3b9","type":"http in","z":"c77db134.66d82","name":"/freeboard/living","url":"/freeboard/living","method":"get","upload":false,"swaggerDoc":"","x":137,"y":261.00000190734863,"wires":[["6e88a2be.4715fc"]]},{"id":"1a70f538.f799db","type":"http in","z":"c77db134.66d82","name":"/freeboard/bedroom","url":"/freeboard/bedroom","method":"get","upload":false,"swaggerDoc":"","x":147,"y":309.00000286102295,"wires":[["faba660a.d4de38"]]},{"id":"95f289f1.9f2568","type":"comment","z":"c77db134.66d82","name":"Freeboard Dashboard - Development Version","info":"*.js & *.css ( embedded images) files \nare embedded into a single index.html ","x":225.1666717529297,"y":78.7638931274414,"wires":[]},{"id":"b7a50570.0bc418","type":"comment","z":"c77db134.66d82","name":"Data Sources Gneration ","info":"","x":150.1666717529297,"y":171.77779293060303,"wires":[]},{"id":"f2fe075f.2e6928","type":"function","z":"c77db134.66d82","name":"Randome Generator","func":"msg.payload = { value : Math.floor(Math.random() * 30) + 60   };\nreturn msg;","outputs":1,"noerr":0,"x":386.00000381469727,"y":217.00000190734863,"wires":[["b9badebf.2988f","cc286368.74c28"]]},{"id":"6e88a2be.4715fc","type":"function","z":"c77db134.66d82","name":"Randome Generator","func":"msg.payload = { value : Math.floor(Math.random() * 30) + 60   };\nreturn msg;","outputs":1,"noerr":0,"x":385,"y":261.0000023841858,"wires":[["b9badebf.2988f"]]},{"id":"faba660a.d4de38","type":"function","z":"c77db134.66d82","name":"Randome Generator","func":"msg.payload = { value : Math.floor(Math.random() * 30) + 60   };\nreturn msg;","outputs":1,"noerr":0,"x":387.00000381469727,"y":306.00000286102295,"wires":[["b9badebf.2988f"]]},{"id":"7f8f09b7.717168","type":"http in","z":"c77db134.66d82","name":"http://localhost:1880/freeboarddist","url":"/freeboarddist/index.html","method":"get","upload":false,"swaggerDoc":"","x":187,"y":421,"wires":[["d06730dc.388b2"]]},{"id":"d06730dc.388b2","type":"file in","z":"c77db134.66d82","name":"index.html","filename":"C:\\Users\\sphyun\\Documents\\project.freeboard\\dist\\index.html","format":"utf8","chunk":false,"sendError":false,"x":408.00002670288086,"y":421.8888883590698,"wires":[["b9badebf.2988f"]]},{"id":"b3236544.de6be8","type":"comment","z":"c77db134.66d82","name":"Freeboard Dashboard  - Production Version","info":"*.js & *.css ( embedded images) files \nare embedded into a single index.html ","x":212.00003051757812,"y":379.097225189209,"wires":[]}]

In Action

5. Generate dashboard.json

{"version":1,"allow_edit":false,"plugins":[],"panes":[{"title":"Kitchen","width":1,"row":{"2":1,"3":1},"col":{"2":1,"3":1},"col_width":1,"widgets":[{"type":"text_widget","settings":{"title":"Current","size":"big","value":"datasources[\"Temperature_Kitchen\"][\"value\"]","sparkline":false,"animate":true,"units":"ºF"}},{"type":"gauge","settings":{"value":"datasources[\"Temperature_Kitchen\"][\"value\"]","min_value":0,"max_value":100}}]},{"title":"Bed Room","width":1,"row":{"2":1,"3":1},"col":{"2":2,"3":3},"col_width":1,"widgets":[{"type":"text_widget","settings":{"title":"Current","size":"big","value":"datasources[\"Temperature_Bedroom\"][\"value\"]","sparkline":false,"animate":true,"units":"ºF"}},{"type":"gauge","settings":{"value":"datasources[\"Temperature_Bedroom\"][\"value\"]","min_value":0,"max_value":100}}]},{"title":"Living Room","width":1,"row":{"2":7,"3":1},"col":{"2":1,"3":2},"col_width":1,"widgets":[{"type":"text_widget","settings":{"title":"Current","size":"big","value":"datasources[\"Temperature_Livingroom\"][\"value\"]","sparkline":false,"animate":true,"units":"ºF"}},{"type":"gauge","settings":{"value":"datasources[\"Temperature_Livingroom\"][\"value\"]","min_value":0,"max_value":100}}]},{"width":1,"row":{"2":7,"3":13},"col":{"2":2,"3":2},"col_width":1,"widgets":[{"type":"text_widget","settings":{"size":"regular","value":"datasources[\"Clock\"][\"time_string_value\"]","animate":true}}]}],"datasources":[{"name":"Temperature_Kitchen","type":"JSON","settings":{"url":"/freeboard/kitchen","use_thingproxy":true,"refresh":5,"method":"GET"}},{"name":"Temperature_Livingroom","type":"JSON","settings":{"url":"/freeboard/living","use_thingproxy":true,"refresh":5,"method":"GET"}},{"name":"Temperature_Bedroom","type":"JSON","settings":{"url":"/freeboard/bedroom","use_thingproxy":true,"refresh":5,"method":"GET"}},{"name":"Clock","type":"clock","settings":{"refresh":1}}],"columns":3}

6. Generate index.html (Production Version)

  • { "allow_edit": false} in dashboard.min.json
  • embedding dashboard.min.json using build:template (gunt-processhtml)
    <!-- build:template 
    <% if ( environment === 'dist') { %> 
    <script> 
    var device_id_23456 = 
    <% } %>
    /build -->
    <!-- build:include:dist dashboard.min.json -->
    <!-- /build -->
    <!-- build:template 
    <% if ( environment === 'dist') { %> 
    ;
    </script>
    <% } %>
    /build -->

...

   <% if ( environment === 'dist') { %>
   // Production Mode
                        freeboard.loadDashboard(device_id_23456, 
                                function() {
                                    freeboard.setEditing(false);
                                });                          
  <% } else { %>
   // Development Mode 
...

7. Crow HTTP Handler for Dashboard

  • GET index.html
    CROW_ROUTE(app, "/")
    ([]{
        std::ifstream file("./index.html");
        std::string content((std::istreambuf_iterator<char>(file)), std::istreambuf_iterator<char>());
        return content;
    });
  • GET /freeboard/kitchen Handler
    CROW_ROUTE(app, "/freeboard/kitchen")
    ([]{
        crow::json::wvalue x;
        int value = 0;
 #ifdef UNITTEST
        // DEMO purpose only. 
        int maxTemp = 80;
        int minTemp = 60;       
        
        value = rand()%(maxTemp - minTemp + 1) + minTemp; 
 #else
        value = 65; // Access the database to retreive the actual value fed by other process/thread 
 #endif
        x["value"] = value;
        return x;
    });

Same for other GET /freeboard/living & /freeboard/bedroom Handlers

8. View-Only Dashboard

8. Related Posts

Releases

No releases published

Packages

No packages published