mirror of
https://github.com/node-red/node-red.git
synced 2023-10-10 13:36:53 +02:00
da61fe12d0
Closes #322 - nodes modules can be installed/removed dynamically at runtime - nodes can be enabled/disabled - onpaletteadd/onpaletteremove api added to node definitions - initial implementation of nr-cli
554 lines
20 KiB
JavaScript
554 lines
20 KiB
JavaScript
/**
|
|
* 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.
|
|
**/
|
|
RED.nodes = (function() {
|
|
|
|
var node_defs = {};
|
|
var nodes = [];
|
|
var configNodes = {};
|
|
var links = [];
|
|
var defaultWorkspace;
|
|
var workspaces = {};
|
|
|
|
var registry = (function() {
|
|
var nodeList = [];
|
|
var nodeSets = {};
|
|
var typeToId = {};
|
|
var nodeDefinitions = {};
|
|
|
|
var exports = {
|
|
getNodeList: function() {
|
|
return nodeList;
|
|
},
|
|
setNodeList: function(list) {
|
|
nodeList = [];
|
|
for(var i=0;i<list.length;i++) {
|
|
var ns = list[i];
|
|
exports.addNodeSet(ns);
|
|
}
|
|
},
|
|
addNodeSet: function(ns) {
|
|
ns.added = false;
|
|
nodeSets[ns.id] = ns;
|
|
for (var j=0;j<ns.types.length;j++) {
|
|
typeToId[ns.types[j]] = ns.id;
|
|
}
|
|
nodeList.push(ns);
|
|
},
|
|
removeNodeSet: function(id) {
|
|
var ns = nodeSets[id];
|
|
for (var j=0;j<ns.types.length;j++) {
|
|
if (ns.added) {
|
|
// TODO: too tightly coupled into palette UI
|
|
RED.palette.remove(ns.types[j]);
|
|
var def = nodeDefinitions[ns.types[j]];
|
|
if (def.onpaletteremove && typeof def.onpaletteremove === "function") {
|
|
def.onpaletteremove.call(def);
|
|
}
|
|
}
|
|
delete typeToId[ns.types[j]];
|
|
}
|
|
delete nodeSets[id];
|
|
for (var i=0;i<nodeList.length;i++) {
|
|
if (nodeList[i].id == id) {
|
|
nodeList.splice(i,1);
|
|
break;
|
|
}
|
|
}
|
|
return ns;
|
|
},
|
|
getNodeSet: function(id) {
|
|
return nodeSets[id];
|
|
},
|
|
enableNodeSet: function(id) {
|
|
var ns = nodeSets[id];
|
|
ns.enabled = true;
|
|
for (var j=0;j<ns.types.length;j++) {
|
|
// TODO: too tightly coupled into palette UI
|
|
RED.palette.show(ns.types[j]);
|
|
var def = nodeDefinitions[ns.types[j]];
|
|
if (def.onpaletteadd && typeof def.onpaletteadd === "function") {
|
|
def.onpaletteadd.call(def);
|
|
}
|
|
}
|
|
},
|
|
disableNodeSet: function(id) {
|
|
var ns = nodeSets[id];
|
|
ns.enabled = false;
|
|
for (var j=0;j<ns.types.length;j++) {
|
|
// TODO: too tightly coupled into palette UI
|
|
RED.palette.hide(ns.types[j]);
|
|
var def = nodeDefinitions[ns.types[j]];
|
|
if (def.onpaletteremove && typeof def.onpaletteremove === "function") {
|
|
def.onpaletteremove.call(def);
|
|
}
|
|
}
|
|
},
|
|
registerNodeType: function(nt,def) {
|
|
nodeDefinitions[nt] = def;
|
|
nodeSets[typeToId[nt]].added = true;
|
|
// TODO: too tightly coupled into palette UI
|
|
RED.palette.add(nt,def);
|
|
if (def.onpaletteadd && typeof def.onpaletteadd === "function") {
|
|
def.onpaletteadd.call(def);
|
|
}
|
|
},
|
|
getNodeType: function(nt) {
|
|
return nodeDefinitions[nt];
|
|
}
|
|
}
|
|
return exports;
|
|
})();
|
|
|
|
function getID() {
|
|
return (1+Math.random()*4294967295).toString(16);
|
|
}
|
|
|
|
function addNode(n) {
|
|
if (n._def.category == "config") {
|
|
configNodes[n.id] = n;
|
|
RED.sidebar.config.refresh();
|
|
} else {
|
|
n.dirty = true;
|
|
nodes.push(n);
|
|
var updatedConfigNode = false;
|
|
for (var d in n._def.defaults) {
|
|
if (n._def.defaults.hasOwnProperty(d)) {
|
|
var property = n._def.defaults[d];
|
|
if (property.type) {
|
|
var type = registry.getNodeType(property.type)
|
|
if (type && type.category == "config") {
|
|
var configNode = configNodes[n[d]];
|
|
if (configNode) {
|
|
updatedConfigNode = true;
|
|
configNode.users.push(n);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
if (updatedConfigNode) {
|
|
RED.sidebar.config.refresh();
|
|
}
|
|
}
|
|
}
|
|
function addLink(l) {
|
|
links.push(l);
|
|
}
|
|
function addConfig(c) {
|
|
configNodes[c.id] = c;
|
|
}
|
|
|
|
function getNode(id) {
|
|
if (id in configNodes) {
|
|
return configNodes[id];
|
|
} else {
|
|
for (var n in nodes) {
|
|
if (nodes[n].id == id) {
|
|
return nodes[n];
|
|
}
|
|
}
|
|
}
|
|
return null;
|
|
}
|
|
|
|
function removeNode(id) {
|
|
var removedLinks = [];
|
|
if (id in configNodes) {
|
|
delete configNodes[id];
|
|
RED.sidebar.config.refresh();
|
|
} else {
|
|
var node = getNode(id);
|
|
if (node) {
|
|
nodes.splice(nodes.indexOf(node),1);
|
|
removedLinks = links.filter(function(l) { return (l.source === node) || (l.target === node); });
|
|
removedLinks.map(function(l) {links.splice(links.indexOf(l), 1); });
|
|
}
|
|
var updatedConfigNode = false;
|
|
for (var d in node._def.defaults) {
|
|
if (node._def.defaults.hasOwnProperty(d)) {
|
|
var property = node._def.defaults[d];
|
|
if (property.type) {
|
|
var type = registry.getNodeType(property.type)
|
|
if (type && type.category == "config") {
|
|
var configNode = configNodes[node[d]];
|
|
if (configNode) {
|
|
updatedConfigNode = true;
|
|
var users = configNode.users;
|
|
users.splice(users.indexOf(node),1);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
if (updatedConfigNode) {
|
|
RED.sidebar.config.refresh();
|
|
}
|
|
}
|
|
return removedLinks;
|
|
}
|
|
|
|
function removeLink(l) {
|
|
var index = links.indexOf(l);
|
|
if (index != -1) {
|
|
links.splice(index,1);
|
|
}
|
|
}
|
|
|
|
function refreshValidation() {
|
|
for (var n=0;n<nodes.length;n++) {
|
|
RED.editor.validateNode(nodes[n]);
|
|
}
|
|
}
|
|
|
|
function addWorkspace(ws) {
|
|
workspaces[ws.id] = ws;
|
|
}
|
|
function getWorkspace(id) {
|
|
return workspaces[id];
|
|
}
|
|
function removeWorkspace(id) {
|
|
delete workspaces[id];
|
|
var removedNodes = [];
|
|
var removedLinks = [];
|
|
var n;
|
|
for (n=0;n<nodes.length;n++) {
|
|
var node = nodes[n];
|
|
if (node.z == id) {
|
|
removedNodes.push(node);
|
|
}
|
|
}
|
|
for (n=0;n<removedNodes.length;n++) {
|
|
var rmlinks = removeNode(removedNodes[n].id);
|
|
removedLinks = removedLinks.concat(rmlinks);
|
|
}
|
|
return {nodes:removedNodes,links:removedLinks};
|
|
}
|
|
|
|
function getAllFlowNodes(node) {
|
|
var visited = {};
|
|
visited[node.id] = true;
|
|
var nns = [node];
|
|
var stack = [node];
|
|
while(stack.length !== 0) {
|
|
var n = stack.shift();
|
|
var childLinks = links.filter(function(d) { return (d.source === n) || (d.target === n);});
|
|
for (var i=0;i<childLinks.length;i++) {
|
|
var child = (childLinks[i].source === n)?childLinks[i].target:childLinks[i].source;
|
|
if (!visited[child.id]) {
|
|
visited[child.id] = true;
|
|
nns.push(child);
|
|
stack.push(child);
|
|
}
|
|
}
|
|
}
|
|
return nns;
|
|
}
|
|
|
|
/**
|
|
* Converts a node to an exportable JSON Object
|
|
**/
|
|
function convertNode(n, exportCreds) {
|
|
exportCreds = exportCreds || false;
|
|
var node = {};
|
|
node.id = n.id;
|
|
node.type = n.type;
|
|
for (var d in n._def.defaults) {
|
|
if (n._def.defaults.hasOwnProperty(d)) {
|
|
node[d] = n[d];
|
|
}
|
|
}
|
|
if(exportCreds && n.credentials) {
|
|
node.credentials = {};
|
|
for (var cred in n._def.credentials) {
|
|
if (n._def.credentials.hasOwnProperty(cred)) {
|
|
if (n.credentials[cred] != null) {
|
|
node.credentials[cred] = n.credentials[cred];
|
|
}
|
|
}
|
|
}
|
|
}
|
|
if (n._def.category != "config") {
|
|
node.x = n.x;
|
|
node.y = n.y;
|
|
node.z = n.z;
|
|
node.wires = [];
|
|
for(var i=0;i<n.outputs;i++) {
|
|
node.wires.push([]);
|
|
}
|
|
var wires = links.filter(function(d){return d.source === n;});
|
|
for (var j=0;j<wires.length;j++) {
|
|
var w = wires[j];
|
|
node.wires[w.sourcePort].push(w.target.id);
|
|
}
|
|
}
|
|
return node;
|
|
}
|
|
|
|
/**
|
|
* Converts the current node selection to an exportable JSON Object
|
|
**/
|
|
function createExportableNodeSet(set) {
|
|
var nns = [];
|
|
var exportedConfigNodes = {};
|
|
for (var n=0;n<set.length;n++) {
|
|
var node = set[n].n;
|
|
var convertedNode = RED.nodes.convertNode(node);
|
|
for (var d in node._def.defaults) {
|
|
if (node._def.defaults[d].type && node[d] in configNodes) {
|
|
var confNode = configNodes[node[d]];
|
|
var exportable = registry.getNodeType(node._def.defaults[d].type).exportable;
|
|
if ((exportable == null || exportable)) {
|
|
if (!(node[d] in exportedConfigNodes)) {
|
|
exportedConfigNodes[node[d]] = true;
|
|
nns.unshift(RED.nodes.convertNode(confNode));
|
|
}
|
|
} else {
|
|
convertedNode[d] = "";
|
|
}
|
|
}
|
|
}
|
|
|
|
nns.push(convertedNode);
|
|
}
|
|
return nns;
|
|
}
|
|
|
|
//TODO: rename this (createCompleteNodeSet)
|
|
function createCompleteNodeSet() {
|
|
var nns = [];
|
|
var i;
|
|
for (i in workspaces) {
|
|
if (workspaces.hasOwnProperty(i)) {
|
|
nns.push(workspaces[i]);
|
|
}
|
|
}
|
|
for (i in configNodes) {
|
|
if (configNodes.hasOwnProperty(i)) {
|
|
nns.push(convertNode(configNodes[i], true));
|
|
}
|
|
}
|
|
for (i=0;i<nodes.length;i++) {
|
|
var node = nodes[i];
|
|
nns.push(convertNode(node, true));
|
|
}
|
|
return nns;
|
|
}
|
|
|
|
function importNodes(newNodesObj,createNewIds) {
|
|
try {
|
|
var i;
|
|
var n;
|
|
var newNodes;
|
|
if (typeof newNodesObj === "string") {
|
|
if (newNodesObj === "") {
|
|
return;
|
|
}
|
|
newNodes = JSON.parse(newNodesObj);
|
|
} else {
|
|
newNodes = newNodesObj;
|
|
}
|
|
|
|
if (!$.isArray(newNodes)) {
|
|
newNodes = [newNodes];
|
|
}
|
|
var unknownTypes = [];
|
|
for (i=0;i<newNodes.length;i++) {
|
|
n = newNodes[i];
|
|
// TODO: remove workspace in next release+1
|
|
if (n.type != "workspace" && n.type != "tab" && !registry.getNodeType(n.type)) {
|
|
// TODO: get this UI thing out of here! (see below as well)
|
|
n.name = n.type;
|
|
n.type = "unknown";
|
|
if (unknownTypes.indexOf(n.name)==-1) {
|
|
unknownTypes.push(n.name);
|
|
}
|
|
if (n.x == null && n.y == null) {
|
|
// config node - remove it
|
|
newNodes.splice(i,1);
|
|
i--;
|
|
}
|
|
}
|
|
}
|
|
if (unknownTypes.length > 0) {
|
|
var typeList = "<ul><li>"+unknownTypes.join("</li><li>")+"</li></ul>";
|
|
var type = "type"+(unknownTypes.length > 1?"s":"");
|
|
RED.notify("<strong>Imported unrecognised "+type+":</strong>"+typeList,"error",false,10000);
|
|
//"DO NOT DEPLOY while in this state.<br/>Either, add missing types to Node-RED, restart and then reload page,<br/>or delete unknown "+n.name+", rewire as required, and then deploy.","error");
|
|
}
|
|
|
|
var new_workspaces = [];
|
|
var workspace_map = {};
|
|
|
|
for (i=0;i<newNodes.length;i++) {
|
|
n = newNodes[i];
|
|
// TODO: remove workspace in next release+1
|
|
if (n.type === "workspace" || n.type === "tab") {
|
|
if (n.type === "workspace") {
|
|
n.type = "tab";
|
|
}
|
|
if (defaultWorkspace == null) {
|
|
defaultWorkspace = n;
|
|
}
|
|
if (createNewIds) {
|
|
var nid = getID();
|
|
workspace_map[n.id] = nid;
|
|
n.id = nid;
|
|
}
|
|
addWorkspace(n);
|
|
RED.view.addWorkspace(n);
|
|
new_workspaces.push(n);
|
|
}
|
|
}
|
|
if (defaultWorkspace == null) {
|
|
defaultWorkspace = { type:"tab", id:getID(), label:"Sheet 1" };
|
|
addWorkspace(defaultWorkspace);
|
|
RED.view.addWorkspace(defaultWorkspace);
|
|
new_workspaces.push(defaultWorkspace);
|
|
}
|
|
|
|
var node_map = {};
|
|
var new_nodes = [];
|
|
var new_links = [];
|
|
|
|
for (i=0;i<newNodes.length;i++) {
|
|
n = newNodes[i];
|
|
// TODO: remove workspace in next release+1
|
|
if (n.type !== "workspace" && n.type !== "tab") {
|
|
var def = registry.getNodeType(n.type);
|
|
if (def && def.category == "config") {
|
|
if (!RED.nodes.node(n.id)) {
|
|
var configNode = {id:n.id,type:n.type,users:[]};
|
|
for (var d in def.defaults) {
|
|
if (def.defaults.hasOwnProperty(d)) {
|
|
configNode[d] = n[d];
|
|
}
|
|
}
|
|
configNode.label = def.label;
|
|
configNode._def = def;
|
|
RED.nodes.add(configNode);
|
|
}
|
|
} else {
|
|
var node = {x:n.x,y:n.y,z:n.z,type:0,wires:n.wires,changed:false};
|
|
if (createNewIds) {
|
|
node.z = workspace_map[node.z];
|
|
if (!workspaces[node.z]) {
|
|
node.z = RED.view.getWorkspace();
|
|
}
|
|
node.id = getID();
|
|
} else {
|
|
node.id = n.id;
|
|
if (node.z == null || !workspaces[node.z]) {
|
|
node.z = RED.view.getWorkspace();
|
|
}
|
|
}
|
|
node.type = n.type;
|
|
node._def = def;
|
|
if (!node._def) {
|
|
node._def = {
|
|
color:"#fee",
|
|
defaults: {},
|
|
label: "unknown: "+n.type,
|
|
labelStyle: "node_label_italic",
|
|
outputs: n.outputs||n.wires.length
|
|
}
|
|
}
|
|
node.outputs = n.outputs||node._def.outputs;
|
|
|
|
for (var d2 in node._def.defaults) {
|
|
if (node._def.defaults.hasOwnProperty(d2)) {
|
|
node[d2] = n[d2];
|
|
}
|
|
}
|
|
|
|
addNode(node);
|
|
RED.editor.validateNode(node);
|
|
node_map[n.id] = node;
|
|
new_nodes.push(node);
|
|
}
|
|
}
|
|
}
|
|
for (i=0;i<new_nodes.length;i++) {
|
|
n = new_nodes[i];
|
|
for (var w1=0;w1<n.wires.length;w1++) {
|
|
var wires = (n.wires[w1] instanceof Array)?n.wires[w1]:[n.wires[w1]];
|
|
for (var w2=0;w2<wires.length;w2++) {
|
|
if (wires[w2] in node_map) {
|
|
var link = {source:n,sourcePort:w1,target:node_map[wires[w2]]};
|
|
addLink(link);
|
|
new_links.push(link);
|
|
}
|
|
}
|
|
}
|
|
delete n.wires;
|
|
}
|
|
return [new_nodes,new_links,new_workspaces];
|
|
} catch(error) {
|
|
//TODO: get this UI thing out of here! (see above as well)
|
|
RED.notify("<strong>Error</strong>: "+error,"error");
|
|
return null;
|
|
}
|
|
|
|
}
|
|
|
|
return {
|
|
registry:registry,
|
|
setNodeList: registry.setNodeList,
|
|
|
|
getNodeSet: registry.getNodeSet,
|
|
addNodeSet: registry.addNodeSet,
|
|
removeNodeSet: registry.removeNodeSet,
|
|
enableNodeSet: registry.enableNodeSet,
|
|
disableNodeSet: registry.disableNodeSet,
|
|
|
|
registerType: registry.registerNodeType,
|
|
getType: registry.getNodeType,
|
|
convertNode: convertNode,
|
|
add: addNode,
|
|
addLink: addLink,
|
|
remove: removeNode,
|
|
removeLink: removeLink,
|
|
addWorkspace: addWorkspace,
|
|
removeWorkspace: removeWorkspace,
|
|
workspace: getWorkspace,
|
|
eachNode: function(cb) {
|
|
for (var n=0;n<nodes.length;n++) {
|
|
cb(nodes[n]);
|
|
}
|
|
},
|
|
eachLink: function(cb) {
|
|
for (var l=0;l<links.length;l++) {
|
|
cb(links[l]);
|
|
}
|
|
},
|
|
eachConfig: function(cb) {
|
|
for (var id in configNodes) {
|
|
if (configNodes.hasOwnProperty(id)) {
|
|
cb(configNodes[id]);
|
|
}
|
|
}
|
|
},
|
|
node: getNode,
|
|
import: importNodes,
|
|
refreshValidation: refreshValidation,
|
|
getAllFlowNodes: getAllFlowNodes,
|
|
createExportableNodeSet: createExportableNodeSet,
|
|
createCompleteNodeSet: createCompleteNodeSet,
|
|
id: getID,
|
|
nodes: nodes, // TODO: exposed for d3 vis
|
|
links: links // TODO: exposed for d3 vis
|
|
};
|
|
})();
|