mirror of
https://github.com/node-red/node-red.git
synced 2023-10-10 13:36:53 +02:00
275 lines
8.5 KiB
JavaScript
275 lines
8.5 KiB
JavaScript
/**
|
|
* Copyright 2013, 2015 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 when = require("when");
|
|
var path = require("path");
|
|
var fs = require("fs");
|
|
|
|
var registry = require("./registry");
|
|
var credentials = require("./credentials");
|
|
var flows = require("./flows");
|
|
var Node = require("./Node");
|
|
var log = require("../log");
|
|
|
|
var events = require("../events");
|
|
|
|
var child_process = require('child_process');
|
|
|
|
var settings;
|
|
|
|
/**
|
|
* Registers a node constructor
|
|
* @param type - the string type name
|
|
* @param constructor - the constructor function for this node type
|
|
* @param opts - optional additional options for the node
|
|
*/
|
|
function registerType(type,constructor,opts) {
|
|
if (opts && opts.credentials) {
|
|
credentials.register(type,opts.credentials);
|
|
}
|
|
registry.registerType(type,constructor);
|
|
}
|
|
|
|
/**
|
|
* Called from a Node's constructor function, invokes the super-class
|
|
* constructor and attaches any credentials to the node.
|
|
* @param node the node object being created
|
|
* @param def the instance definition for the node
|
|
*/
|
|
function createNode(node,def) {
|
|
Node.call(node,def);
|
|
var id = node.id;
|
|
if (def._alias) {
|
|
id = def._alias;
|
|
}
|
|
var creds = credentials.get(id);
|
|
if (creds) {
|
|
//console.log("Attaching credentials to ",node.id);
|
|
node.credentials = creds;
|
|
} else if (credentials.getDefinition(node.type)) {
|
|
node.credentials = {};
|
|
}
|
|
}
|
|
|
|
function init(_settings,storage) {
|
|
settings = _settings;
|
|
credentials.init(storage);
|
|
flows.init(_settings,storage);
|
|
registry.init(_settings);
|
|
}
|
|
|
|
function checkTypeInUse(id) {
|
|
var nodeInfo = registry.getNodeInfo(id);
|
|
if (!nodeInfo) {
|
|
throw new Error(log._("nodes.index.unrecognised-id", {id:id}));
|
|
} else {
|
|
var inUse = {};
|
|
var config = flows.getFlows();
|
|
config.forEach(function(n) {
|
|
inUse[n.type] = (inUse[n.type]||0)+1;
|
|
});
|
|
var nodesInUse = [];
|
|
nodeInfo.types.forEach(function(t) {
|
|
if (inUse[t]) {
|
|
nodesInUse.push(t);
|
|
}
|
|
});
|
|
if (nodesInUse.length > 0) {
|
|
var msg = nodesInUse.join(", ");
|
|
var err = new Error(log._("nodes.index.type-in-use", {msg:msg}));
|
|
err.code = "type_in_use";
|
|
throw err;
|
|
}
|
|
}
|
|
}
|
|
|
|
function removeNode(id) {
|
|
checkTypeInUse(id);
|
|
return registry.removeNode(id);
|
|
}
|
|
|
|
function removeModule(module) {
|
|
var info = registry.getModuleInfo(module);
|
|
if (!info) {
|
|
throw new Error(log._("nodes.index.unrecognised-module", {module:module}));
|
|
} else {
|
|
for (var i=0;i<info.nodes.length;i++) {
|
|
checkTypeInUse(module+"/"+info.nodes[i].name);
|
|
}
|
|
return registry.removeModule(module);
|
|
}
|
|
}
|
|
|
|
function disableNode(id) {
|
|
checkTypeInUse(id);
|
|
return registry.disableNode(id);
|
|
}
|
|
|
|
function installModule(module) {
|
|
//TODO: ensure module is 'safe'
|
|
return when.promise(function(resolve,reject) {
|
|
if (/[\s;]/.test(module)) {
|
|
reject(new Error(log._("server.install.invalid")));
|
|
return;
|
|
}
|
|
if (registry.getModuleInfo(module)) {
|
|
// TODO: nls
|
|
var err = new Error("Module already loaded");
|
|
err.code = "module_already_loaded";
|
|
reject(err);
|
|
return;
|
|
}
|
|
log.info(log._("server.install.installing",{name: module}));
|
|
|
|
var installDir = settings.userDir || process.env.NODE_RED_HOME || ".";
|
|
var child = child_process.exec('npm install --production '+module,
|
|
{
|
|
cwd: installDir
|
|
},
|
|
function(err, stdin, stdout) {
|
|
if (err) {
|
|
var lookFor404 = new RegExp(" 404 .*"+module+"$","m");
|
|
if (lookFor404.test(stdout)) {
|
|
log.warn(log._("server.install.install-failed-not-found",{name:module}));
|
|
var e = new Error();
|
|
e.code = 404;
|
|
reject(e);
|
|
} else {
|
|
log.warn(log._("server.install.install-failed-long",{name:module}));
|
|
log.warn("------------------------------------------");
|
|
log.warn(err.toString());
|
|
log.warn("------------------------------------------");
|
|
reject(new Error(log._("server.install.install-failed")));
|
|
}
|
|
} else {
|
|
log.info(log._("server.install.installed",{name:module}));
|
|
resolve(registry.addModule(module).then(reportAddedModules));
|
|
}
|
|
}
|
|
);
|
|
});
|
|
}
|
|
|
|
|
|
function reportAddedModules(info) {
|
|
//comms.publish("node/added",info.nodes,false);
|
|
if (info.nodes.length > 0) {
|
|
log.info(log._("server.added-types"));
|
|
for (var i=0;i<info.nodes.length;i++) {
|
|
for (var j=0;j<info.nodes[i].types.length;j++) {
|
|
log.info(" - "+
|
|
(info.nodes[i].module?info.nodes[i].module+":":"")+
|
|
info.nodes[i].types[j]+
|
|
(info.nodes[i].err?" : "+info.nodes[i].err:"")
|
|
);
|
|
}
|
|
}
|
|
}
|
|
return info;
|
|
}
|
|
|
|
function reportRemovedModules(removedNodes) {
|
|
//comms.publish("node/removed",removedNodes,false);
|
|
log.info(log._("server.removed-types"));
|
|
for (var j=0;j<removedNodes.length;j++) {
|
|
for (var i=0;i<removedNodes[j].types.length;i++) {
|
|
log.info(" - "+(removedNodes[j].module?removedNodes[j].module+":":"")+removedNodes[j].types[i]);
|
|
}
|
|
}
|
|
return removedNodes;
|
|
}
|
|
|
|
function uninstallModule(module) {
|
|
return when.promise(function(resolve,reject) {
|
|
if (/[\s;]/.test(module)) {
|
|
reject(new Error(log._("server.install.invalid")));
|
|
return;
|
|
}
|
|
var installDir = settings.userDir || process.env.NODE_RED_HOME || ".";
|
|
var moduleDir = path.join(installDir,"node_modules",module);
|
|
if (!fs.existsSync(moduleDir)) {
|
|
return reject(new Error(log._("server.install.uninstall-failed",{name:module})));
|
|
}
|
|
|
|
var list = removeModule(module);
|
|
log.info(log._("server.install.uninstalling",{name:module}));
|
|
var child = child_process.exec('npm remove '+module,
|
|
{
|
|
cwd: installDir
|
|
},
|
|
function(err, stdin, stdout) {
|
|
if (err) {
|
|
log.warn(log._("server.install.uninstall-failed-long",{name:module}));
|
|
log.warn("------------------------------------------");
|
|
log.warn(err.toString());
|
|
log.warn("------------------------------------------");
|
|
reject(new Error(log._("server.install.uninstall-failed",{name:module})));
|
|
} else {
|
|
log.info(log._("server.install.uninstalled",{name:module}));
|
|
reportRemovedModules(list);
|
|
resolve(list);
|
|
}
|
|
}
|
|
);
|
|
});
|
|
}
|
|
|
|
|
|
module.exports = {
|
|
// Lifecycle
|
|
init: init,
|
|
load: registry.load,
|
|
|
|
// Node registry
|
|
createNode: createNode,
|
|
getNode: flows.get,
|
|
eachNode: flows.eachNode,
|
|
|
|
installModule: installModule,
|
|
uninstallModule: uninstallModule,
|
|
|
|
enableNode: registry.enableNode,
|
|
disableNode: disableNode,
|
|
|
|
// Node type registry
|
|
registerType: registerType,
|
|
getType: registry.get,
|
|
|
|
getNodeInfo: registry.getNodeInfo,
|
|
getNodeList: registry.getNodeList,
|
|
|
|
getModuleInfo: registry.getModuleInfo,
|
|
|
|
getNodeConfigs: registry.getNodeConfigs,
|
|
getNodeConfig: registry.getNodeConfig,
|
|
|
|
clearRegistry: registry.clear,
|
|
cleanModuleList: registry.cleanModuleList,
|
|
|
|
// Flow handling
|
|
loadFlows: flows.load,
|
|
startFlows: flows.startFlows,
|
|
stopFlows: flows.stopFlows,
|
|
setFlows: flows.setFlows,
|
|
getFlows: flows.getFlows,
|
|
|
|
// Credentials
|
|
addCredentials: credentials.add,
|
|
getCredentials: credentials.get,
|
|
deleteCredentials: credentials.delete,
|
|
getCredentialDefinition: credentials.getDefinition
|
|
};
|