2013-09-05 15:02:48 +01:00
|
|
|
/**
|
|
|
|
* Copyright 2013 IBM Corp.
|
|
|
|
*
|
|
|
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
* you may not use this file except in compliance with the License.
|
|
|
|
* You may obtain a copy of the License at
|
|
|
|
*
|
|
|
|
* http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
*
|
|
|
|
* Unless required by applicable law or agreed to in writing, software
|
|
|
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
* See the License for the specific language governing permissions and
|
|
|
|
* limitations under the License.
|
|
|
|
**/
|
|
|
|
var util = require("util");
|
2013-09-09 20:03:22 +01:00
|
|
|
var EventEmitter = require("events").EventEmitter;
|
2013-09-05 15:02:48 +01:00
|
|
|
var fs = require("fs");
|
2013-11-13 17:02:29 +00:00
|
|
|
var path = require("path");
|
2013-11-21 14:03:17 +00:00
|
|
|
var clone = require("clone");
|
2013-09-09 20:03:22 +01:00
|
|
|
var events = require("./events");
|
2013-11-10 00:05:58 +00:00
|
|
|
var storage = null;
|
2013-09-05 15:02:48 +01:00
|
|
|
|
|
|
|
function getCallerFilename(type) {
|
|
|
|
//if (type == "summary") {
|
|
|
|
// var err = new Error();
|
|
|
|
// console.log(err.stack);
|
|
|
|
// return null;
|
|
|
|
//}
|
|
|
|
// Save original Error.prepareStackTrace
|
|
|
|
var origPrepareStackTrace = Error.prepareStackTrace;
|
|
|
|
// Override with function that just returns `stack`
|
|
|
|
Error.prepareStackTrace = function (_, stack) {
|
|
|
|
return stack;
|
|
|
|
}
|
|
|
|
// Create a new `Error`, which automatically gets `stack`
|
|
|
|
var err = new Error();
|
|
|
|
// Evaluate `err.stack`, which calls our new `Error.prepareStackTrace`
|
|
|
|
var stack = err.stack;
|
|
|
|
// Restore original `Error.prepareStackTrace`
|
|
|
|
Error.prepareStackTrace = origPrepareStackTrace;
|
|
|
|
// Remove superfluous function call on stack
|
|
|
|
stack.shift();
|
|
|
|
stack.shift();
|
|
|
|
return stack[0].getFileName();
|
|
|
|
}
|
|
|
|
|
|
|
|
var registry = (function() {
|
|
|
|
var nodes = {};
|
|
|
|
var logHandlers = [];
|
|
|
|
var obj = {
|
|
|
|
add: function(n) {
|
|
|
|
nodes[n.id] = n;
|
|
|
|
n.on("log",function(msg) {
|
|
|
|
for (var i in logHandlers) {
|
|
|
|
logHandlers[i].emit("log",msg);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
},
|
|
|
|
get: function(i) {
|
|
|
|
return nodes[i];
|
|
|
|
},
|
|
|
|
clear: function() {
|
2013-09-09 20:03:22 +01:00
|
|
|
events.emit("nodes-stopping");
|
2013-09-05 15:02:48 +01:00
|
|
|
for (var n in nodes) {
|
|
|
|
nodes[n].close();
|
|
|
|
}
|
2013-09-09 20:03:22 +01:00
|
|
|
events.emit("nodes-stopped");
|
2013-09-05 15:02:48 +01:00
|
|
|
nodes = {};
|
|
|
|
},
|
2013-09-20 17:15:45 +01:00
|
|
|
each: function(cb) {
|
|
|
|
for (var n in nodes) {
|
|
|
|
cb(nodes[n]);
|
|
|
|
}
|
|
|
|
},
|
2013-09-05 15:02:48 +01:00
|
|
|
addLogHandler: function(handler) {
|
|
|
|
logHandlers.push(handler);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return obj;
|
|
|
|
})();
|
|
|
|
|
2013-09-09 20:03:22 +01:00
|
|
|
var ConsoleLogHandler = new EventEmitter();
|
2013-09-05 15:02:48 +01:00
|
|
|
ConsoleLogHandler.on("log",function(msg) {
|
|
|
|
util.log("["+msg.level+"] ["+msg.type+":"+(msg.name||msg.id)+"] "+msg.msg);
|
|
|
|
});
|
|
|
|
|
|
|
|
registry.addLogHandler(ConsoleLogHandler);
|
|
|
|
|
|
|
|
var node_type_registry = (function() {
|
|
|
|
var node_types = {};
|
|
|
|
var node_configs = {};
|
|
|
|
var obj = {
|
|
|
|
register: function(type,node) {
|
|
|
|
util.inherits(node, Node);
|
|
|
|
var callerFilename = getCallerFilename(type);
|
|
|
|
if (callerFilename == null) {
|
|
|
|
util.log("["+type+"] unable to determine filename");
|
|
|
|
} else {
|
|
|
|
var configFilename = callerFilename.replace(/\.js$/,".html");
|
|
|
|
if (fs.existsSync(configFilename)) {
|
|
|
|
node_types[type] = node;
|
|
|
|
if (! node_configs[configFilename]) {
|
|
|
|
node_configs[configFilename] = fs.readFileSync(configFilename,'utf8');
|
|
|
|
}
|
2013-09-18 21:12:59 +01:00
|
|
|
events.emit("type-registered",type);
|
2013-09-05 15:02:48 +01:00
|
|
|
} else {
|
|
|
|
util.log("["+type+"] missing template file: "+configFilename);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
},
|
|
|
|
get: function(type) {
|
|
|
|
return node_types[type];
|
|
|
|
},
|
|
|
|
getNodeConfigs: function() {
|
|
|
|
var result = "";
|
|
|
|
for (var nt in node_configs) {
|
|
|
|
result += node_configs[nt];
|
|
|
|
}
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return obj;
|
|
|
|
})();
|
|
|
|
|
|
|
|
function Node(n) {
|
|
|
|
this.id = n.id;
|
|
|
|
registry.add(this);
|
|
|
|
this.type = n.type;
|
|
|
|
if (n.name) {
|
|
|
|
this.name = n.name;
|
|
|
|
}
|
|
|
|
this.wires = n.wires||[];
|
|
|
|
}
|
2013-09-09 20:03:22 +01:00
|
|
|
util.inherits(Node,EventEmitter);
|
2013-09-05 15:02:48 +01:00
|
|
|
|
|
|
|
Node.prototype.close = function() {
|
|
|
|
// called when a node is removed
|
2013-09-20 17:15:45 +01:00
|
|
|
this.emit("close");
|
2013-09-05 15:02:48 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
Node.prototype.send = function(msg) {
|
|
|
|
// instanceof doesn't work for some reason here
|
|
|
|
if (msg == null) {
|
|
|
|
msg = [];
|
|
|
|
} else if (!util.isArray(msg)) {
|
|
|
|
msg = [msg];
|
|
|
|
}
|
|
|
|
for (var i in this.wires) {
|
|
|
|
var wires = this.wires[i];
|
|
|
|
if (i < msg.length) {
|
2013-11-21 14:03:17 +00:00
|
|
|
if (msg[i] != null) {
|
|
|
|
var msgs = msg[i];
|
|
|
|
if (!util.isArray(msg[i])) {
|
|
|
|
msgs = [msg[i]];
|
|
|
|
}
|
2013-11-29 19:56:46 +00:00
|
|
|
//if (wires.length == 1) {
|
|
|
|
// // Single recipient, don't need to clone the message
|
|
|
|
// var node = registry.get(wires[0]);
|
|
|
|
// if (node) {
|
|
|
|
// for (var k in msgs) {
|
|
|
|
// var mm = msgs[k];
|
|
|
|
// node.receive(mm);
|
|
|
|
// }
|
|
|
|
// }
|
|
|
|
//} else {
|
2013-11-21 14:03:17 +00:00
|
|
|
// Multiple recipients, must send message copies
|
|
|
|
for (var j in wires) {
|
2013-09-05 15:02:48 +01:00
|
|
|
var node = registry.get(wires[j]);
|
|
|
|
if (node) {
|
2013-11-21 14:03:17 +00:00
|
|
|
for (var k in msgs) {
|
|
|
|
var mm = msgs[k];
|
2013-11-28 16:06:17 +00:00
|
|
|
// Temporary fix for #97
|
|
|
|
// TODO: remove this http-node-specific fix somehow
|
|
|
|
var req = mm.req;
|
|
|
|
var res = mm.res;
|
2013-12-05 14:39:26 +00:00
|
|
|
delete mm.req;
|
|
|
|
delete mm.res;
|
2013-11-21 14:03:17 +00:00
|
|
|
var m = clone(mm);
|
2013-11-28 16:06:17 +00:00
|
|
|
m.req = req;
|
|
|
|
m.res = res;
|
2013-12-05 14:39:26 +00:00
|
|
|
mm.req = req;
|
|
|
|
mm.res = res;
|
2013-11-21 14:03:17 +00:00
|
|
|
node.receive(m);
|
|
|
|
}
|
2013-09-05 15:02:48 +01:00
|
|
|
}
|
|
|
|
}
|
2013-11-29 19:56:46 +00:00
|
|
|
//}
|
2013-09-05 15:02:48 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
module.exports.Node = Node;
|
|
|
|
|
|
|
|
Node.prototype.receive = function(msg) {
|
|
|
|
this.emit("input",msg);
|
|
|
|
}
|
|
|
|
|
|
|
|
Node.prototype.log = function(msg) {
|
|
|
|
var o = {level:'log',id:this.id, type:this.type, msg:msg};
|
|
|
|
if (this.name) o.name = this.name;
|
|
|
|
this.emit("log",o);
|
|
|
|
}
|
|
|
|
Node.prototype.warn = function(msg) {
|
|
|
|
var o = {level:'warn',id:this.id, type:this.type, msg:msg};
|
|
|
|
if (this.name) o.name = this.name;
|
|
|
|
this.emit("log",o);
|
|
|
|
}
|
|
|
|
Node.prototype.error = function(msg) {
|
|
|
|
var o = {level:'error',id:this.id, type:this.type, msg:msg};
|
|
|
|
if (this.name) o.name = this.name;
|
|
|
|
this.emit("log",o);
|
|
|
|
}
|
|
|
|
|
|
|
|
var credentials = {};
|
|
|
|
|
|
|
|
module.exports.addCredentials = function(id,creds) {
|
|
|
|
credentials[id] = creds;
|
2013-11-22 13:53:34 +00:00
|
|
|
if (!storage) {
|
|
|
|
// Do this lazily to ensure the storage provider as been initialised
|
|
|
|
storage = require("./storage");
|
|
|
|
}
|
2013-11-10 00:05:58 +00:00
|
|
|
storage.saveCredentials(credentials);
|
2013-09-05 15:02:48 +01:00
|
|
|
}
|
|
|
|
module.exports.getCredentials = function(id) {
|
|
|
|
return credentials[id];
|
|
|
|
}
|
|
|
|
module.exports.deleteCredentials = function(id) {
|
|
|
|
delete credentials[id];
|
2013-11-10 00:05:58 +00:00
|
|
|
storage.saveCredentials(credentials);
|
2013-09-05 15:02:48 +01:00
|
|
|
}
|
|
|
|
module.exports.createNode = function(node,def) {
|
|
|
|
Node.call(node,def);
|
|
|
|
}
|
|
|
|
|
|
|
|
module.exports.registerType = node_type_registry.register;
|
|
|
|
module.exports.getNodeConfigs = node_type_registry.getNodeConfigs;
|
|
|
|
module.exports.addLogHandler = registry.addLogHandler;
|
|
|
|
|
2013-11-13 17:02:29 +00:00
|
|
|
module.exports.load = function(settings) {
|
2013-09-05 15:02:48 +01:00
|
|
|
function loadNodes(dir) {
|
|
|
|
fs.readdirSync(dir).sort().filter(function(fn){
|
2013-11-13 17:02:29 +00:00
|
|
|
var stats = fs.statSync(path.join(dir,fn));
|
2013-09-05 15:02:48 +01:00
|
|
|
if (stats.isFile()) {
|
|
|
|
if (/\.js$/.test(fn)) {
|
|
|
|
try {
|
2013-11-13 17:02:29 +00:00
|
|
|
require(path.join(dir,fn));
|
2013-09-05 15:02:48 +01:00
|
|
|
} catch(err) {
|
|
|
|
util.log("["+fn+"] "+err);
|
|
|
|
//console.log(err.stack);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} else if (stats.isDirectory()) {
|
2013-11-20 13:55:21 +00:00
|
|
|
// Ignore /.dirs/, /lib/ /node_modules/
|
|
|
|
if (!/^(\..*|lib|icons|node_modules)$/.test(fn)) {
|
2013-11-13 17:02:29 +00:00
|
|
|
loadNodes(path.join(dir,fn));
|
2013-10-13 21:01:46 +01:00
|
|
|
} else if (fn === "icons") {
|
2013-11-13 17:02:29 +00:00
|
|
|
events.emit("node-icon-dir",path.join(dir,fn));
|
2013-09-05 15:02:48 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
2013-09-26 21:13:09 +01:00
|
|
|
loadNodes(__dirname+"/../nodes");
|
2013-11-13 17:02:29 +00:00
|
|
|
if (settings.nodesDir) {
|
|
|
|
loadNodes(settings.nodesDir);
|
|
|
|
}
|
2013-09-18 21:12:59 +01:00
|
|
|
//events.emit("nodes-loaded");
|
2013-09-05 15:02:48 +01:00
|
|
|
}
|
|
|
|
|
2013-11-10 00:05:58 +00:00
|
|
|
var activeConfig = [];
|
2013-09-18 21:12:59 +01:00
|
|
|
var missingTypes = [];
|
|
|
|
|
|
|
|
events.on('type-registered',function(type) {
|
|
|
|
if (missingTypes.length > 0) {
|
|
|
|
var i = missingTypes.indexOf(type);
|
|
|
|
if (i != -1) {
|
|
|
|
missingTypes.splice(i,1);
|
|
|
|
util.log("[red] Missing type registered: "+type);
|
|
|
|
}
|
|
|
|
if (missingTypes.length == 0) {
|
|
|
|
parseConfig();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
2013-09-05 15:02:48 +01:00
|
|
|
module.exports.getNode = function(nid) {
|
|
|
|
return registry.get(nid);
|
|
|
|
}
|
2013-10-12 22:00:34 +01:00
|
|
|
|
2013-10-13 19:14:39 +01:00
|
|
|
function stopFlows() {
|
2013-09-18 21:12:59 +01:00
|
|
|
if (activeConfig&&activeConfig.length > 0) {
|
|
|
|
util.log("[red] Stopping flows");
|
|
|
|
}
|
2013-09-05 15:02:48 +01:00
|
|
|
registry.clear();
|
2013-10-13 19:14:39 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
module.exports.stopFlows = stopFlows;
|
|
|
|
|
|
|
|
module.exports.setConfig = function(conf) {
|
|
|
|
stopFlows();
|
2013-09-18 21:12:59 +01:00
|
|
|
activeConfig = conf;
|
2013-11-10 00:05:58 +00:00
|
|
|
|
|
|
|
if (!storage) {
|
|
|
|
// Do this lazily to ensure the storage provider as been initialised
|
2013-11-12 17:13:06 +00:00
|
|
|
storage = require("./storage");
|
2013-11-10 00:05:58 +00:00
|
|
|
}
|
|
|
|
storage.getCredentials().then(function(creds) {
|
|
|
|
credentials = creds;
|
|
|
|
parseConfig();
|
|
|
|
}).otherwise(function(err) {
|
|
|
|
util.log("[red] Error loading credentials : "+err);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
module.exports.getConfig = function() {
|
|
|
|
return activeConfig;
|
2013-09-18 21:12:59 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
var parseConfig = function() {
|
|
|
|
missingTypes = [];
|
|
|
|
for (var i in activeConfig) {
|
|
|
|
var type = activeConfig[i].type;
|
2013-10-30 21:45:45 +00:00
|
|
|
// TODO: remove workspace in next release+1
|
|
|
|
if (type != "workspace" && type != "tab") {
|
2013-10-25 21:34:00 +01:00
|
|
|
var nt = node_type_registry.get(type);
|
|
|
|
if (!nt && missingTypes.indexOf(type) == -1) {
|
|
|
|
missingTypes.push(type);
|
|
|
|
}
|
2013-09-18 21:12:59 +01:00
|
|
|
}
|
|
|
|
};
|
|
|
|
if (missingTypes.length > 0) {
|
|
|
|
util.log("[red] Waiting for missing types to be registered:");
|
|
|
|
for (var i in missingTypes) {
|
|
|
|
util.log("[red] - "+missingTypes[i]);
|
|
|
|
}
|
|
|
|
return;
|
|
|
|
}
|
2013-09-13 17:23:23 +01:00
|
|
|
|
2013-09-18 21:12:59 +01:00
|
|
|
util.log("[red] Starting flows");
|
2013-09-09 20:03:22 +01:00
|
|
|
events.emit("nodes-starting");
|
2013-09-18 21:12:59 +01:00
|
|
|
for (var i in activeConfig) {
|
2013-09-05 15:02:48 +01:00
|
|
|
var nn = null;
|
2013-10-30 21:45:45 +00:00
|
|
|
// TODO: remove workspace in next release+1
|
|
|
|
if (activeConfig[i].type != "workspace" && activeConfig[i].type != "tab") {
|
2013-10-25 21:34:00 +01:00
|
|
|
var nt = node_type_registry.get(activeConfig[i].type);
|
|
|
|
if (nt) {
|
|
|
|
try {
|
|
|
|
nn = new nt(activeConfig[i]);
|
|
|
|
}
|
|
|
|
catch (err) {
|
|
|
|
util.log("[red] "+activeConfig[i].type+" : "+err);
|
|
|
|
}
|
2013-10-12 22:00:34 +01:00
|
|
|
}
|
2013-10-25 21:34:00 +01:00
|
|
|
// console.log(nn);
|
|
|
|
if (nn == null) {
|
|
|
|
util.log("[red] unknown type: "+activeConfig[i].type);
|
2013-10-12 22:00:34 +01:00
|
|
|
}
|
2013-09-05 15:02:48 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
// Clean up any orphaned credentials
|
|
|
|
var deletedCredentials = false;
|
|
|
|
for (var c in credentials) {
|
|
|
|
var n = registry.get(c);
|
|
|
|
if (!n) {
|
|
|
|
deletedCredentials = true;
|
|
|
|
delete credentials[c];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (deletedCredentials) {
|
2013-11-10 00:05:58 +00:00
|
|
|
storage.saveCredentials(credentials);
|
2013-09-05 15:02:48 +01:00
|
|
|
}
|
2013-09-09 20:03:22 +01:00
|
|
|
events.emit("nodes-started");
|
2013-09-05 15:02:48 +01:00
|
|
|
}
|