Add node status updates

This commit is contained in:
Nick O'Leary 2014-05-08 14:15:54 +01:00
parent 8e7fc011f0
commit 7ecb80bf40
7 changed files with 145 additions and 42 deletions

View File

@ -303,6 +303,7 @@
<script src="orion/built-editor.min.js"></script> <script src="orion/built-editor.min.js"></script>
<script src="d3.v3.min.js"></script> <script src="d3.v3.min.js"></script>
<script src="red/main.js"></script> <script src="red/main.js"></script>
<script src="red/comms.js"></script>
<script src="red/ui/state.js"></script> <script src="red/ui/state.js"></script>
<script src="red/nodes.js"></script> <script src="red/nodes.js"></script>
<script src="red/history.js"></script> <script src="red/history.js"></script>
@ -317,6 +318,5 @@
<script src="red/ui/editor.js"></script> <script src="red/ui/editor.js"></script>
<script src="red/ui/library.js"></script> <script src="red/ui/library.js"></script>
<script src="red/ui/notifications.js"></script> <script src="red/ui/notifications.js"></script>
<script src="red/comms.js"></script>
</body> </body>
</html> </html>

View File

@ -18,24 +18,34 @@ RED.comms = function() {
var errornotification = null; var errornotification = null;
var subscriptions = {}; var subscriptions = {};
var ws;
function connectWS() { function connectWS() {
var path = location.hostname+":"+location.port+document.location.pathname; var path = location.hostname+":"+location.port+document.location.pathname;
path = path+(path.slice(-1) == "/"?"":"/")+"comms"; path = path+(path.slice(-1) == "/"?"":"/")+"comms";
path = "ws"+(document.location.protocol=="https:"?"s":"")+"://"+path; path = "ws"+(document.location.protocol=="https:"?"s":"")+"://"+path;
var ws = new WebSocket(path); ws = new WebSocket(path);
ws.onopen = function() { ws.onopen = function() {
if (errornotification) { if (errornotification) {
errornotification.close(); errornotification.close();
errornotification = null; errornotification = null;
} }
for (var t in subscriptions) {
ws.send(JSON.stringify({subscribe:t}));
}
} }
ws.onmessage = function(event) { ws.onmessage = function(event) {
var msg = JSON.parse(event.data); var msg = JSON.parse(event.data);
var subscribers = subscriptions[msg.topic]; if (msg.topic) {
if (subscribers) { for (var t in subscriptions) {
for (var i=0;i<subscribers.length;i++) { var re = new RegExp("^"+t.replace(/([\[\]\?\(\)\\\\$\^\*\.|])/g,"\\$1").replace(/\+/g,"[^/]+").replace(/\/#$/,"(\/.*)?")+"$");
subscribers[i](msg.topic,msg.data); if (re.test(msg.topic)) {
var subscribers = subscriptions[t];
if (subscribers) {
for (var i=0;i<subscribers.length;i++) {
subscribers[i](msg.topic,msg.data);
}
}
}
} }
} }
}; };
@ -46,16 +56,20 @@ RED.comms = function() {
setTimeout(connectWS,1000); setTimeout(connectWS,1000);
} }
} }
connectWS();
function subscribe(topic,callback) { function subscribe(topic,callback) {
if (subscriptions[topic] == null) { if (subscriptions[topic] == null) {
subscriptions[topic] = []; subscriptions[topic] = [];
} }
subscriptions[topic].push(callback); subscriptions[topic].push(callback);
if (ws && ws.readyState == 1) {
ws.send(JSON.stringify({subscribe:topic}));
}
} }
return { return {
connect: connectWS,
subscribe: subscribe subscribe: subscribe
} }
}(); }();

View File

@ -52,13 +52,13 @@ var RED = function() {
var invalid = false; var invalid = false;
var unknownNodes = []; var unknownNodes = [];
RED.nodes.eachNode(function(node) { RED.nodes.eachNode(function(node) {
invalid = invalid || !node.valid; invalid = invalid || !node.valid;
if (node.type === "unknown") { if (node.type === "unknown") {
if (unknownNodes.indexOf(node.name) == -1) { if (unknownNodes.indexOf(node.name) == -1) {
unknownNodes.push(node.name); unknownNodes.push(node.name);
}
invalid = true;
} }
invalid = true;
}
}); });
if (invalid) { if (invalid) {
if (unknownNodes.length > 0) { if (unknownNodes.length > 0) {
@ -137,25 +137,34 @@ var RED = function() {
function loadSettings() { function loadSettings() {
$.get('settings', function(data) { $.get('settings', function(data) {
RED.settings = data; RED.settings = data;
loadNodes(); loadNodes();
}); });
} }
function loadNodes() { function loadNodes() {
$.get('nodes', function(data) { $.get('nodes', function(data) {
$("body").append(data); $("body").append(data);
$(".palette-spinner").hide(); $(".palette-spinner").hide();
$(".palette-scroll").show(); $(".palette-scroll").show();
$("#palette-search").show(); $("#palette-search").show();
loadFlows(); loadFlows();
}); });
} }
function loadFlows() { function loadFlows() {
$.getJSON("flows",function(nodes) { $.getJSON("flows",function(nodes) {
RED.nodes.import(nodes); RED.nodes.import(nodes);
RED.view.dirty(false); RED.view.dirty(false);
RED.view.redraw(); RED.view.redraw();
RED.comms.subscribe("status/#",function(topic,msg) {
var parts = topic.split("/");
var node = RED.nodes.node(parts[1]);
if (node) {
node.status = msg;
node.dirty = true;
RED.view.redraw();
}
});
}); });
} }
@ -168,18 +177,19 @@ var RED = function() {
//}); //});
dialog.on('show',function() { dialog.on('show',function() {
RED.keyboard.disable(); RED.keyboard.disable();
}); });
dialog.on('hidden',function() { dialog.on('hidden',function() {
RED.keyboard.enable(); RED.keyboard.enable();
}); });
dialog.modal(); dialog.modal();
} }
$(function() { $(function() {
RED.keyboard.add(/* ? */ 191,{shift:true},function(){showHelp();d3.event.preventDefault();}); RED.keyboard.add(/* ? */ 191,{shift:true},function(){showHelp();d3.event.preventDefault();});
loadSettings(); loadSettings();
RED.comms.connect();
}); });
return { return {

View File

@ -44,6 +44,12 @@ RED.view = function() {
var clipboard = ""; var clipboard = "";
var status_colours = {
"red" : "#00c",
"green": "#5a8"
}
var outer = d3.select("#chart") var outer = d3.select("#chart")
.append("svg:svg") .append("svg:svg")
.attr("width", space_width) .attr("width", space_width)
@ -811,7 +817,7 @@ RED.view = function() {
.on("click",function(d) { d._def.onbadgeclick.call(d);d3.event.preventDefault();}); .on("click",function(d) { d._def.onbadgeclick.call(d);d3.event.preventDefault();});
} }
} }
if (d._def.button) { if (d._def.button) {
var nodeButtonGroup = node.append('svg:g') var nodeButtonGroup = node.append('svg:g')
.attr("transform",function(d) { return "translate("+((d._def.align == "right") ? 94 : -25)+",2)"; }) .attr("transform",function(d) { return "translate("+((d._def.align == "right") ? 94 : -25)+",2)"; })
@ -893,6 +899,21 @@ RED.view = function() {
text.attr('text-anchor','end'); text.attr('text-anchor','end');
} }
var status = node.append("svg:g").attr("class","node_status_group").style("display","none");
var statusRect = status.append("circle").attr("class","node_status")
.attr("cx",9).attr("cy",6).attr("r",5).attr("stroke-width","3");
var statusLabel = status.append("svg:text")
.attr("class","node_status_label")
.attr('x',20).attr('y',10)
.style({
'stroke-width': 0,
'fill': '#999',
'font-size':'9pt',
'stroke':'#000',
'text-anchor':'start'
});
//node.append("circle").attr({"class":"centerDot","cx":0,"cy":0,"r":5}); //node.append("circle").attr({"class":"centerDot","cx":0,"cy":0,"r":5});
if (d._def.inputs > 0) { if (d._def.inputs > 0) {
@ -1017,7 +1038,37 @@ RED.view = function() {
} }
return ""; return "";
}); });
if (!d.status) {
thisNode.selectAll('.node_status_group').style("display","none");
} else {
thisNode.selectAll('.node_status_group').style("display","inline").attr("transform","translate(3,"+(d.h+3)+")");
var fill = status_colours[d.status.fill]; // Only allow our colours for now
if (d.status.shape == null && fill == null) {
thisNode.selectAll('.node_status').style("display","none");
} else {
var style;
if (d.status.shape == null || d.status.shape == "dot") {
style = {
display: "inline",
fill: fill,
stroke: fill
};
} else if (d.status.shape == "ring" ){
style = {
display: "inline",
fill: '#fff',
stroke: fill
}
}
thisNode.selectAll('.node_status').style(style);
}
if (d.status.text) {
thisNode.selectAll('.node_status_label').text(d.status.text);
} else {
thisNode.selectAll('.node_status_label').text("");
}
}
d.dirty = false; d.dirty = false;
} }
}); });

View File

@ -23,6 +23,8 @@ var settings;
var wsServer; var wsServer;
var activeConnections = []; var activeConnections = [];
var retained = {};
var heartbeatTimer; var heartbeatTimer;
var lastSentTime; var lastSentTime;
@ -47,6 +49,12 @@ function start() {
} }
} }
}); });
ws.on('message', function(data,flags) {
var msg = JSON.parse(data);
if (msg.subscribe) {
handleRemoteSubscription(ws,msg.subscribe);
}
});
ws.on('error', function(err) { ws.on('error', function(err) {
util.log("[red:comms] error : "+err.toString()); util.log("[red:comms] error : "+err.toString());
}); });
@ -67,17 +75,34 @@ function start() {
}, 15000); }, 15000);
} }
function publish(topic,data) { function publish(topic,data,retain) {
var msg = JSON.stringify({topic:topic,data:data}); if (retain) {
retained[topic] = data;
}
lastSentTime = Date.now();
activeConnections.forEach(function(conn) { activeConnections.forEach(function(conn) {
try { publishTo(conn,topic,data);
conn.send(msg);
} catch(err) {
util.log("[red:comms] send error : "+err.toString());
}
}); });
} }
function publishTo(ws,topic,data) {
var msg = JSON.stringify({topic:topic,data:data});
try {
ws.send(msg);
} catch(err) {
util.log("[red:comms] send error : "+err.toString());
}
}
function handleRemoteSubscription(ws,topic) {
var re = new RegExp("^"+topic.replace(/([\[\]\?\(\)\\\\$\^\*\.|])/g,"\\$1").replace(/\+/g,"[^/]+").replace(/\/#$/,"(\/.*)?")+"$");
for (var t in retained) {
if (re.test(t)) {
publishTo(ws,t,retained[t]);
}
}
}
module.exports = { module.exports = {
init:init, init:init,

View File

@ -20,7 +20,7 @@ var clone = require("clone");
var flows = require("./flows"); var flows = require("./flows");
var comms = require("../comms");
function Node(n) { function Node(n) {
this.id = n.id; this.id = n.id;
@ -114,6 +114,11 @@ Node.prototype.error = function(msg) {
if (this.name) o.name = this.name; if (this.name) o.name = this.name;
this.emit("log",o); this.emit("log",o);
} }
/**
* status: { fill:"red|green", shape:"dot|ring", text:"blah" }
*/
Node.prototype.status = function(status,retain) {
comms.publish("status/"+this.id,status,retain);
}
module.exports = Node; module.exports = Node;

View File

@ -18,7 +18,6 @@ var events = require("./events");
var server = require("./server"); var server = require("./server");
var nodes = require("./nodes"); var nodes = require("./nodes");
var library = require("./library"); var library = require("./library");
var comms = require("./comms");
var log = require("./log"); var log = require("./log");
var fs = require("fs"); var fs = require("fs");
var settings = null; var settings = null;
@ -51,7 +50,6 @@ var RED = {
stop: server.stop, stop: server.stop,
nodes: nodes, nodes: nodes,
library: library, library: library,
comms: comms,
events: events, events: events,
log: log log: log
}; };