added justgage for display of power consumption
added power consumption in spaceapi
This commit is contained in:
parent
156616ebcf
commit
0dd746c47f
10 changed files with 193 additions and 64 deletions
34
node/bla
34
node/bla
|
@ -1,34 +0,0 @@
|
||||||
member: cn=tidirium,ou=people,dc=chaostreff-dortmund,dc=de
|
|
||||||
member: cn=tatonka,ou=people,dc=chaostreff-dortmund,dc=de
|
|
||||||
member: cn=ulihd,ou=people,dc=chaostreff-dortmund,dc=de
|
|
||||||
member: cn=nospoonuser,ou=people,dc=chaostreff-dortmund,dc=de
|
|
||||||
member: cn=progamler,ou=people,dc=chaostreff-dortmund,dc=de
|
|
||||||
member: cn=zeus,ou=people,dc=chaostreff-dortmund,dc=de
|
|
||||||
member: cn=lichtscheu,ou=people,dc=chaostreff-dortmund,dc=de
|
|
||||||
member: cn=stefan,ou=people,dc=chaostreff-dortmund,dc=de
|
|
||||||
member: cn=wegginho,ou=people,dc=chaostreff-dortmund,dc=de
|
|
||||||
member: cn=jcoder,ou=people,dc=chaostreff-dortmund,dc=de
|
|
||||||
member: cn=smash,ou=people,dc=chaostreff-dortmund,dc=de
|
|
||||||
member: cn=pascal,ou=people,dc=chaostreff-dortmund,dc=de
|
|
||||||
member: cn=herbern,ou=people,dc=chaostreff-dortmund,dc=de
|
|
||||||
member: cn=syn,ou=people,dc=chaostreff-dortmund,dc=de
|
|
||||||
member: cn=seppo,ou=people,dc=chaostreff-dortmund,dc=de
|
|
||||||
member: cn=blueberry,ou=people,dc=chaostreff-dortmund,dc=de
|
|
||||||
member: cn=immo,ou=people,dc=chaostreff-dortmund,dc=de
|
|
||||||
member: cn=freakfairy,ou=people,dc=chaostreff-dortmund,dc=de
|
|
||||||
member: cn=badboy,ou=people,dc=chaostreff-dortmund,dc=de
|
|
||||||
member: cn=alteisen,ou=people,dc=chaostreff-dortmund,dc=de
|
|
||||||
member: cn=eimann,ou=people,dc=chaostreff-dortmund,dc=de
|
|
||||||
member: cn=downhill,ou=people,dc=chaostreff-dortmund,dc=de
|
|
||||||
member: cn=annett,ou=people,dc=chaostreff-dortmund,dc=de
|
|
||||||
member: cn=dukat,ou=people,dc=chaostreff-dortmund,dc=de
|
|
||||||
member: cn=klapauzius,ou=people,dc=chaostreff-dortmund,dc=de
|
|
||||||
member: cn=hotte,ou=people,dc=chaostreff-dortmund,dc=de
|
|
||||||
member: cn=henne,ou=people,dc=chaostreff-dortmund,dc=de
|
|
||||||
member: cn=tunix,ou=people,dc=chaostreff-dortmund,dc=de
|
|
||||||
member: cn=fiurin,ou=people,dc=chaostreff-dortmund,dc=de
|
|
||||||
member: cn=jsilence,ou=people,dc=chaostreff-dortmund,dc=de
|
|
||||||
member: cn=kuro,ou=people,dc=chaostreff-dortmund,dc=de
|
|
||||||
member: cn=klara,ou=people,dc=chaostreff-dortmund,dc=de
|
|
||||||
member: cn=lucas,ou=people,dc=chaostreff-dortmund,dc=de
|
|
||||||
|
|
65
node/flukso.js
Normal file
65
node/flukso.js
Normal file
|
@ -0,0 +1,65 @@
|
||||||
|
var redis = require("redis");
|
||||||
|
var util = require('util');
|
||||||
|
var EventEmitter = require('events').EventEmitter;
|
||||||
|
var http = require("http");
|
||||||
|
|
||||||
|
var redisprefix = "flukso:";
|
||||||
|
|
||||||
|
var Flukso = function(hostname, pathname) {
|
||||||
|
var self = this;
|
||||||
|
|
||||||
|
var redisClient = redis.createClient();
|
||||||
|
var regexp = /([0-9]+)\]\]$/; // /\(([0-9]+) hosts* up\)/;
|
||||||
|
|
||||||
|
|
||||||
|
redisClient.on("connect", function () {
|
||||||
|
console.log("connected to redis");
|
||||||
|
self.emit('ready');
|
||||||
|
});
|
||||||
|
|
||||||
|
this.pollPower = function() {
|
||||||
|
|
||||||
|
http.request({ host: hostname, path: pathname }, function(response) {
|
||||||
|
var str = '';
|
||||||
|
|
||||||
|
response.on('data', function(chunk) {
|
||||||
|
str += chunk;
|
||||||
|
});
|
||||||
|
|
||||||
|
response.on('end', function () {
|
||||||
|
// console.log("http received: " + str);
|
||||||
|
|
||||||
|
var matches = regexp.exec(str);
|
||||||
|
if(matches != null && matches.length == 2) {
|
||||||
|
var time = Date.now();
|
||||||
|
|
||||||
|
redisClient.zremrangebyscore(redisprefix + 'power', "-inf", time - 7*24*60*1000);
|
||||||
|
|
||||||
|
var num = matches[1];
|
||||||
|
|
||||||
|
redisClient.zadd(redisprefix + 'power', time, time + "|" + num, function() {
|
||||||
|
self.emit('done', parseInt(num));
|
||||||
|
});
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}).end();
|
||||||
|
};
|
||||||
|
|
||||||
|
// this.getHistory = function(start, end, callback) {
|
||||||
|
//
|
||||||
|
// redisClient.zrangebyscore(redisprefix + 'power', start, end, function(err, replies) {
|
||||||
|
// var data = [];
|
||||||
|
//
|
||||||
|
// replies.forEach(function (reply, i) {
|
||||||
|
// var line = reply.split('|');
|
||||||
|
// data.push( { at: moment(parseInt(line[0])).format(), value: line[1] });
|
||||||
|
// });
|
||||||
|
//
|
||||||
|
// callback(data);
|
||||||
|
// });
|
||||||
|
// };
|
||||||
|
};
|
||||||
|
|
||||||
|
util.inherits(Flukso, EventEmitter);
|
||||||
|
|
||||||
|
module.exports = Flukso;
|
|
@ -10,7 +10,7 @@ var IpPoll = function(switchaddr, hostsaddr) {
|
||||||
|
|
||||||
var redisClient = redis.createClient();
|
var redisClient = redis.createClient();
|
||||||
var regexp = /\(([0-9]+) hosts* up\)/;
|
var regexp = /\(([0-9]+) hosts* up\)/;
|
||||||
var nmap = "nmap -n -sP -T4 ";
|
var nmap = "nmap -n -sP -T3 ";
|
||||||
|
|
||||||
redisClient.on("connect", function () {
|
redisClient.on("connect", function () {
|
||||||
console.log("connected to redis");
|
console.log("connected to redis");
|
||||||
|
@ -24,12 +24,12 @@ var IpPoll = function(switchaddr, hostsaddr) {
|
||||||
if(matches != null && matches.length == 2) {
|
if(matches != null && matches.length == 2) {
|
||||||
var time = Date.now();
|
var time = Date.now();
|
||||||
|
|
||||||
redisClient.zremrangebyscore('onlinecount', "-inf", time - 7*24*60*1000);
|
redisClient.zremrangebyscore(redisprefix + 'onlinecount', "-inf", time - 7*24*60*1000);
|
||||||
|
|
||||||
var num = matches[1];
|
var num = matches[1];
|
||||||
|
|
||||||
redisClient.zadd('onlinecount', time, time + "|" + num, function() {
|
redisClient.zadd(redisprefix + 'onlinecount', time, time + "|" + num, function() {
|
||||||
self.emit('doneCount', num);
|
self.emit('doneCount', parseInt(num));
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -51,7 +51,7 @@ var IpPoll = function(switchaddr, hostsaddr) {
|
||||||
|
|
||||||
this.getHistory = function(start, end, callback) {
|
this.getHistory = function(start, end, callback) {
|
||||||
|
|
||||||
redisClient.zrangebyscore('onlinecount', "-inf", "+inf", function(err, replies) {
|
redisClient.zrangebyscore(redisprefix + 'onlinecount', "-inf", "+inf", function(err, replies) {
|
||||||
var data = [];
|
var data = [];
|
||||||
|
|
||||||
replies.forEach(function (reply, i) {
|
replies.forEach(function (reply, i) {
|
||||||
|
|
|
@ -1,3 +1,11 @@
|
||||||
|
body {
|
||||||
|
margin: 20px 0 0 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
.intro {
|
||||||
|
margin: 10px 0 40px 0;
|
||||||
|
}
|
||||||
|
|
||||||
.status-icon {
|
.status-icon {
|
||||||
width: 128px;
|
width: 128px;
|
||||||
height: 128px;
|
height: 128px;
|
||||||
|
@ -28,3 +36,10 @@ h2 {
|
||||||
height: 200px;
|
height: 200px;
|
||||||
margin-top: 0;
|
margin-top: 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#gauge {
|
||||||
|
width: 290px;
|
||||||
|
height: 200px;
|
||||||
|
display: inline-block;
|
||||||
|
margin: 0;
|
||||||
|
}
|
|
@ -1,5 +1,7 @@
|
||||||
'use strict';
|
'use strict';
|
||||||
|
|
||||||
|
var gpower;
|
||||||
|
|
||||||
angular.module('roomstateapp.controllers', []).
|
angular.module('roomstateapp.controllers', []).
|
||||||
controller('StatusCtrl', function ($scope, $http, Socket) {
|
controller('StatusCtrl', function ($scope, $http, Socket) {
|
||||||
|
|
||||||
|
@ -9,6 +11,7 @@ angular.module('roomstateapp.controllers', []).
|
||||||
}).
|
}).
|
||||||
success(function (data, status, headers, config) {
|
success(function (data, status, headers, config) {
|
||||||
$scope.simple = data;
|
$scope.simple = data;
|
||||||
|
gpower.refresh(message.data.power);
|
||||||
}).
|
}).
|
||||||
error(function (data, status, headers, config) {
|
error(function (data, status, headers, config) {
|
||||||
//$scope.name = 'Error!'
|
//$scope.name = 'Error!'
|
||||||
|
@ -18,6 +21,7 @@ angular.module('roomstateapp.controllers', []).
|
||||||
Socket.on('sdata', function(message) {
|
Socket.on('sdata', function(message) {
|
||||||
console.log("received data from server: " + message.data.names);
|
console.log("received data from server: " + message.data.names);
|
||||||
$scope.simple = message.data;
|
$scope.simple = message.data;
|
||||||
|
gpower.refresh(message.data.power);
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
|
@ -64,4 +68,15 @@ angular.module('roomstateapp.statusfilter', []).filter('statustostring', functio
|
||||||
angular.module('roomstateapp', ['roomstateapp.controllers', 'roomstateapp.services', 'roomstateapp.statusfilter']);
|
angular.module('roomstateapp', ['roomstateapp.controllers', 'roomstateapp.services', 'roomstateapp.statusfilter']);
|
||||||
|
|
||||||
|
|
||||||
|
$( document ).ready(function() {
|
||||||
|
console.log( "ready!" );
|
||||||
|
|
||||||
|
gpower = new JustGage({
|
||||||
|
id: "gauge",
|
||||||
|
value: 0,
|
||||||
|
min: 0,
|
||||||
|
max: 15000,
|
||||||
|
title: "Power"
|
||||||
|
});
|
||||||
|
|
||||||
|
});
|
||||||
|
|
12
node/public/js/vendor/justgage.1.0.1.min.js
vendored
Normal file
12
node/public/js/vendor/justgage.1.0.1.min.js
vendored
Normal file
File diff suppressed because one or more lines are too long
10
node/public/js/vendor/raphael.2.1.0.min.js
vendored
Normal file
10
node/public/js/vendor/raphael.2.1.0.min.js
vendored
Normal file
File diff suppressed because one or more lines are too long
|
@ -6,6 +6,7 @@ var flash = require('connect-flash');
|
||||||
var moment = require('moment');
|
var moment = require('moment');
|
||||||
var SnmpMac = require("./snmp-mac");
|
var SnmpMac = require("./snmp-mac");
|
||||||
var IpPoll = require("./ip-poll");
|
var IpPoll = require("./ip-poll");
|
||||||
|
var Flukso = require("./flukso");
|
||||||
var routes = require("./routes");
|
var routes = require("./routes");
|
||||||
var path = require('path');
|
var path = require('path');
|
||||||
|
|
||||||
|
@ -33,14 +34,29 @@ var spaceanswer = {
|
||||||
issue_report_channels: [ "issue_mail" ],
|
issue_report_channels: [ "issue_mail" ],
|
||||||
state: {
|
state: {
|
||||||
open: null,
|
open: null,
|
||||||
lastchange: 0
|
lastchange: 0,
|
||||||
|
icon: {
|
||||||
|
open: "http://status.ctdo.de/img/green.png",
|
||||||
|
closed: "http://status.ctdo.de/img/red.png"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
sensors: {
|
||||||
|
power_consumption: [ {
|
||||||
|
unit: "W",
|
||||||
|
name: "mains power",
|
||||||
|
description: "overall power consumption from our space",
|
||||||
|
location: "basement",
|
||||||
|
value: 0 } ]
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
var simpleanswer = { state: "unknown", count: 0, names: [], lastchange: 0 };
|
var simpleanswer = { state: "unknown", count: 0, names: [], lastchange: 0, power: 0 };
|
||||||
var usercountanswer = { at: 0, current_value: 0, datapoints: [ { at : 0, value: 0 } ]};
|
var usercountanswer = { at: 0, current_value: 0, datapoints: [
|
||||||
|
{ at: 0, value: 0 }
|
||||||
|
]};
|
||||||
|
|
||||||
var snmpMac = new SnmpMac("juni.ctdo.de", "ctdo23");
|
var snmpMac = new SnmpMac("juni.ctdo.de", "ctdo23");
|
||||||
var ippoll = new IpPoll("switch2.raum.ctdo.de", "195.160.169.20-62 195.160.169.70-126");
|
var ippoll = new IpPoll("switch2.raum.ctdo.de", "195.160.169.20-62 195.160.169.70-126");
|
||||||
|
var flukso = new Flukso("flukso.raum.ctdo.de", "/sensor/cf00e0b22230f4a8870af58f2b8719dd");
|
||||||
|
|
||||||
snmpMac.on('done', function (res) {
|
snmpMac.on('done', function (res) {
|
||||||
simpleanswer.names = res;
|
simpleanswer.names = res;
|
||||||
|
@ -61,6 +77,12 @@ ippoll.on('doneState', function (state) {
|
||||||
io.sockets.emit('sdata', { data: simpleanswer });
|
io.sockets.emit('sdata', { data: simpleanswer });
|
||||||
});
|
});
|
||||||
|
|
||||||
|
flukso.on('done', function (num) {
|
||||||
|
simpleanswer.power = num;
|
||||||
|
spaceanswer.sensors.power_consumption[0].value = num;
|
||||||
|
io.sockets.emit('sdata', { data: simpleanswer });
|
||||||
|
});
|
||||||
|
|
||||||
io.configure(function () {
|
io.configure(function () {
|
||||||
io.set('log level', 0);
|
io.set('log level', 0);
|
||||||
});
|
});
|
||||||
|
@ -69,17 +91,14 @@ io.configure(function () {
|
||||||
function work() {
|
function work() {
|
||||||
snmpMac.poll();
|
snmpMac.poll();
|
||||||
ippoll.pollCount();
|
ippoll.pollCount();
|
||||||
|
|
||||||
// simpleanswer.names.length = 0;
|
|
||||||
// for(var i=0;i<4+Math.random() * 100;i++) {
|
|
||||||
// simpleanswer.names.push("nickname"+i);
|
|
||||||
// }
|
|
||||||
// io.sockets.emit('sdata', { "data": simpleanswer }); // TODO: enable data simulation in dev env
|
|
||||||
}
|
}
|
||||||
|
|
||||||
setInterval(work, 60000); // TODO: make 60s in prod and 10s in dev env
|
setInterval(work, 60000);
|
||||||
work();
|
work();
|
||||||
|
|
||||||
|
setInterval(flukso.pollPower, 5000);
|
||||||
|
flukso.pollPower();
|
||||||
|
|
||||||
app.set('snmpMac', snmpMac); //TODO: wie kann man das schoener machen? (Modul in die Routes reintun)
|
app.set('snmpMac', snmpMac); //TODO: wie kann man das schoener machen? (Modul in die Routes reintun)
|
||||||
app.set('views', __dirname + '/views');
|
app.set('views', __dirname + '/views');
|
||||||
app.set('view engine', 'jade');
|
app.set('view engine', 'jade');
|
||||||
|
@ -97,12 +116,19 @@ app.use(flash());
|
||||||
|
|
||||||
|
|
||||||
app.get('/api/spaceapi/v13', function (req, res) {
|
app.get('/api/spaceapi/v13', function (req, res) {
|
||||||
|
res.setHeader("Access-Control-Allow-Origin", "*");
|
||||||
|
res.setHeader("Cache-Control", "no-cache");
|
||||||
res.send(spaceanswer);
|
res.send(spaceanswer);
|
||||||
});
|
});
|
||||||
app.get('/api/simple/v2', function (req, res) {
|
app.get('/api/simple/v2', function (req, res) {
|
||||||
|
res.setHeader("Access-Control-Allow-Origin", "*");
|
||||||
|
res.setHeader("Cache-Control", "no-cache");
|
||||||
res.send(simpleanswer);
|
res.send(simpleanswer);
|
||||||
});
|
});
|
||||||
app.get('/api/usercount', function (req, res) {
|
app.get('/api/usercount', function (req, res) {
|
||||||
|
res.setHeader("Access-Control-Allow-Origin", "*");
|
||||||
|
res.setHeader("Cache-Control", "no-cache");
|
||||||
|
|
||||||
//TODO: respect query params "start", "end", "interval" (s) and "limit" (like Xively)
|
//TODO: respect query params "start", "end", "interval" (s) and "limit" (like Xively)
|
||||||
// maybe skip "interval" if code gets too complex :)
|
// maybe skip "interval" if code gets too complex :)
|
||||||
|
|
||||||
|
@ -116,6 +142,9 @@ app.get('/api/usercount', function (req, res) {
|
||||||
|
|
||||||
});
|
});
|
||||||
app.get('/api/simple/image', function (req, res) {
|
app.get('/api/simple/image', function (req, res) {
|
||||||
|
res.setHeader("Access-Control-Allow-Origin", "*");
|
||||||
|
res.setHeader("Cache-Control", "no-cache");
|
||||||
|
|
||||||
if (spaceanswer.state.open == true) {
|
if (spaceanswer.state.open == true) {
|
||||||
res.sendfile(path.resolve(__dirname, 'public/img/green.png'));
|
res.sendfile(path.resolve(__dirname, 'public/img/green.png'));
|
||||||
} else if (spaceanswer.state.open == false) {
|
} else if (spaceanswer.state.open == false) {
|
||||||
|
@ -130,7 +159,7 @@ app.post('/form', routes.form);
|
||||||
app.get('/', routes.index);
|
app.get('/', routes.index);
|
||||||
|
|
||||||
|
|
||||||
server.listen(3000, 'localhost');
|
server.listen(3000, '0.0.0.0');
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
|
@ -4,7 +4,7 @@ block content
|
||||||
|
|
||||||
.ink-grid(ng-controller='StatusCtrl')
|
.ink-grid(ng-controller='StatusCtrl')
|
||||||
.column-group.gutters
|
.column-group.gutters
|
||||||
.large-20.small-100
|
.large-20.medium-20.small-100
|
||||||
|
|
||||||
|
|
||||||
div(class="status-icon status-icon-{{simple.state}}")
|
div(class="status-icon status-icon-{{simple.state}}")
|
||||||
|
@ -21,7 +21,7 @@ block content
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
.large-80.small-100
|
.large-50.medium-80.small-100
|
||||||
h2 Anzahl Geräte:
|
h2 Anzahl Geräte:
|
||||||
|
|
||||||
#graph
|
#graph
|
||||||
|
@ -36,9 +36,18 @@ block content
|
||||||
span(ng-show=" ! $last ")
|
span(ng-show=" ! $last ")
|
||||||
| , <!-- -->
|
| , <!-- -->
|
||||||
|
|
||||||
|
.large-30.medium-100.small-100
|
||||||
|
h2 Energieverbrauch
|
||||||
|
|
||||||
|
#gauge.power
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
block scripts
|
block scripts
|
||||||
script(type="text/javascript",src="/js/vendor/d3.min.js")
|
script(type="text/javascript",src="/js/vendor/d3.min.js")
|
||||||
script(type="text/javascript",src="/js/vendor/d3.layout.min.js")
|
script(type="text/javascript",src="/js/vendor/d3.layout.min.js")
|
||||||
script(type="text/javascript",src="/js/vendor/rickshaw.min.js")
|
script(type="text/javascript",src="/js/vendor/rickshaw.min.js")
|
||||||
script(type="text/javascript",src="/js/vendor/moment.min.js")
|
script(type="text/javascript",src="/js/vendor/moment.min.js")
|
||||||
script(type="text/javascript",src="/js/graph.js")
|
script(type="text/javascript",src="/js/graph.js")
|
||||||
|
script(type="text/javascript",src="/js/vendor/raphael.2.1.0.min.js")
|
||||||
|
script(type="text/javascript",src="/js/vendor/justgage.1.0.1.min.js")
|
||||||
|
|
|
@ -11,6 +11,7 @@ html(ng-app="roomstateapp")
|
||||||
link(rel='stylesheet', href='/css/ink-min.css')
|
link(rel='stylesheet', href='/css/ink-min.css')
|
||||||
link(rel='stylesheet', href='/css/rickshaw.min.css')
|
link(rel='stylesheet', href='/css/rickshaw.min.css')
|
||||||
link(rel='stylesheet', href='/css/style.css')
|
link(rel='stylesheet', href='/css/style.css')
|
||||||
|
meta(name="viewport", content="width=device-width, initial-scale=1.0, maximum-scale=1.0, user-scalable=0")
|
||||||
body
|
body
|
||||||
|
|
||||||
.ink-grid
|
.ink-grid
|
||||||
|
@ -18,6 +19,13 @@ html(ng-app="roomstateapp")
|
||||||
header
|
header
|
||||||
h1 chaostreff status
|
h1 chaostreff status
|
||||||
|
|
||||||
|
.intro
|
||||||
|
p
|
||||||
|
| Hier siehst du den aktuellen Status von uns. Die Daten werden vom Router und unserem Flukso
|
||||||
|
| eingesammelt. Die Anzahl der Geräte und der Status selber werden Minütlich abgefragt. Unseren
|
||||||
|
| Energieverbrauch bekommst du alle fünf Sekunden neu. Den Status "geöffnet" oder "geschlossen"
|
||||||
|
| bestimmt das Programm mit einem ping auf den Switch im Raum.
|
||||||
|
|
||||||
.main-content
|
.main-content
|
||||||
block content
|
block content
|
||||||
|
|
||||||
|
|
Loading…
Reference in a new issue