node-red/red/runtime/index.js

262 lines
9.1 KiB
JavaScript
Raw Normal View History

2013-09-05 16:02:48 +02:00
/**
* Copyright JS Foundation and other contributors, http://js.foundation
2013-09-05 16:02:48 +02:00
*
* 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');
2013-09-05 16:02:48 +02:00
var redNodes = require("./nodes");
2014-09-22 15:33:26 +02:00
var storage = require("./storage");
var library = require("./library");
var events = require("./events");
var settings = require("./settings");
var express = require("express");
var path = require('path');
var fs = require("fs");
2016-02-22 18:47:16 +01:00
var os = require("os");
2013-09-05 16:02:48 +02:00
var redUtil;
var log;
var i18n;
2015-02-04 23:28:17 +01:00
var runtimeMetricInterval = null;
var started = false;
2015-11-24 23:38:42 +01:00
var stubbedExpressApp = {
get: function() {},
post: function() {},
put: function() {},
delete: function() {}
2015-11-24 23:38:42 +01:00
}
var adminApi = {
auth: {
needsPermission: function() {}
2015-11-24 23:38:42 +01:00
},
comms: {
publish: function() {}
2015-11-24 23:38:42 +01:00
},
adminApp: stubbedExpressApp,
server: {}
}
var nodeApp;
function init(userSettings,_redUtil,_adminApi) {
redUtil = _redUtil;
log = redUtil.log;
i18n = redUtil.i18n;
userSettings.version = getVersion();
settings.init(userSettings);
nodeApp = express();
2015-11-24 23:38:42 +01:00
if (_adminApi) {
adminApi = _adminApi;
}
redNodes.init(runtime);
library.init(runtime);
}
2015-02-04 23:28:17 +01:00
var version;
function getVersion() {
if (!version) {
version = require(path.join(__dirname,"..","..","package.json")).version;
/* istanbul ignore else */
try {
fs.statSync(path.join(__dirname,"..","..",".git"));
version += "-git";
} catch(err) {
// No git directory
}
}
return version;
2014-11-04 12:34:49 +01:00
}
function start() {
2018-07-30 00:47:19 +02:00
return i18n.registerMessageCatalog("runtime",path.resolve(path.join(__dirname,"locales")),"runtime.json")
.then(function() { return storage.init(runtime)})
.then(function() { return settings.load(storage)})
.then(function() {
2015-02-04 23:28:17 +01:00
if (log.metric()) {
runtimeMetricInterval = setInterval(function() {
reportMetrics();
2015-03-21 18:42:06 +01:00
}, settings.runtimeMetricInterval||15000);
2015-02-04 23:28:17 +01:00
}
log.info("\n\n"+log._("runtime.welcome")+"\n===================\n");
2014-11-04 12:34:49 +01:00
if (settings.version) {
2015-05-08 15:21:01 +02:00
log.info(log._("runtime.version",{component:"Node-RED",version:"v"+settings.version}));
}
2015-05-08 15:21:01 +02:00
log.info(log._("runtime.version",{component:"Node.js ",version:process.version}));
if (settings.UNSUPPORTED_VERSION) {
log.error("*****************************************************************");
log.error("* "+log._("runtime.unsupported_version",{component:"Node.js",version:process.version,requires: ">=4"})+" *");
log.error("*****************************************************************");
events.emit("runtime-event",{id:"runtime-unsupported-version",payload:{type:"error",text:"notification.errors.unsupportedVersion"},retain:true});
}
2016-02-22 18:47:16 +01:00
log.info(os.type()+" "+os.release()+" "+os.arch()+" "+os.endianness());
return redNodes.load().then(function() {
2014-11-04 12:34:49 +01:00
var i;
var nodeErrors = redNodes.getNodeList(function(n) { return n.err!=null;});
var nodeMissing = redNodes.getNodeList(function(n) { return n.module && n.enabled && !n.loaded && !n.err;});
2014-11-04 12:34:49 +01:00
if (nodeErrors.length > 0) {
log.warn("------------------------------------------------------");
for (i=0;i<nodeErrors.length;i+=1) {
if (nodeErrors[i].err.code === "type_already_registered") {
log.warn("["+nodeErrors[i].id+"] "+log._("server.type-already-registered",{type:nodeErrors[i].err.details.type,module: nodeErrors[i].err.details.moduleA}));
} else {
log.warn("["+nodeErrors[i].id+"] "+nodeErrors[i].err);
}
}
log.warn("------------------------------------------------------");
2014-09-22 15:33:26 +02:00
}
2014-11-04 12:34:49 +01:00
if (nodeMissing.length > 0) {
2015-05-08 15:21:01 +02:00
log.warn(log._("server.missing-modules"));
2014-11-04 12:34:49 +01:00
var missingModules = {};
for (i=0;i<nodeMissing.length;i++) {
var missing = nodeMissing[i];
missingModules[missing.module] = missingModules[missing.module]||{
module:missing.module,
version:missing.pending_version||missing.version,
types:[]
}
missingModules[missing.module].types = missingModules[missing.module].types.concat(missing.types);
2014-11-04 12:34:49 +01:00
}
var moduleList = [];
2014-11-04 12:34:49 +01:00
var promises = [];
var installingModules = [];
2014-11-04 12:34:49 +01:00
for (i in missingModules) {
if (missingModules.hasOwnProperty(i)) {
log.warn(" - "+i+" ("+missingModules[i].version+"): "+missingModules[i].types.join(", "));
if (settings.autoInstallModules && i != "node-red") {
installingModules.push({id:i,version:missingModules[i].version});
2014-09-22 15:33:26 +02:00
}
}
}
2014-11-04 12:34:49 +01:00
if (!settings.autoInstallModules) {
2015-05-08 15:21:01 +02:00
log.info(log._("server.removing-modules"));
redNodes.cleanModuleList();
} else if (installingModules.length > 0) {
reinstallAttempts = 0;
reinstallModules(installingModules);
2014-11-04 12:34:49 +01:00
}
}
2015-12-07 00:29:58 +01:00
if (settings.settingsFile) {
log.info(log._("runtime.paths.settings",{path:settings.settingsFile}));
}
if (settings.httpStatic) {
log.info(log._("runtime.paths.httpStatic",{path:path.resolve(settings.httpStatic)}));
}
2018-07-20 16:26:47 +02:00
redNodes.loadContextsPlugin().then(function () {
redNodes.loadFlows().then(redNodes.startFlows).catch(function(err) {});
started = true;
});
2017-10-17 00:23:50 +02:00
}).catch(function(err) {
2014-11-04 12:34:49 +01:00
console.log(err);
});
});
}
2014-11-04 12:34:49 +01:00
var reinstallAttempts;
var reinstallTimeout;
function reinstallModules(moduleList) {
var promises = [];
var failedModules = [];
for (var i=0;i<moduleList.length;i++) {
if (settings.autoInstallModules && i != "node-red") {
promises.push(redNodes.installModule(moduleList[i].id,moduleList[i].version));
}
}
when.settle(promises).then(function(results) {
var reinstallList = [];
for (var i=0;i<results.length;i++) {
if (results[i].state === 'rejected') {
reinstallList.push(moduleList[i]);
} else {
events.emit("runtime-event",{id:"node/added",retain:false,payload:results[i].value.nodes});
}
}
if (reinstallList.length > 0) {
reinstallAttempts++;
// First 5 at 1x timeout, next 5 at 2x, next 5 at 4x, then 8x
var timeout = (settings.autoInstallModulesRetry||30000) * Math.pow(2,Math.min(Math.floor(reinstallAttempts/5),3));
reinstallTimeout = setTimeout(function() {
reinstallModules(reinstallList);
},timeout);
}
});
}
2015-02-04 23:28:17 +01:00
function reportMetrics() {
var memUsage = process.memoryUsage();
2015-03-21 18:42:06 +01:00
log.log({
level: log.METRIC,
event: "runtime.memory.rss",
value: memUsage.rss
});
log.log({
level: log.METRIC,
event: "runtime.memory.heapTotal",
value: memUsage.heapTotal
});
log.log({
level: log.METRIC,
event: "runtime.memory.heapUsed",
value: memUsage.heapUsed
});
2015-02-04 23:28:17 +01:00
}
function stop() {
2015-02-04 23:28:17 +01:00
if (runtimeMetricInterval) {
clearInterval(runtimeMetricInterval);
runtimeMetricInterval = null;
}
if (reinstallTimeout) {
clearTimeout(reinstallTimeout);
}
started = false;
2018-05-30 03:24:27 +02:00
return redNodes.stopFlows().then(function(){
return redNodes.closeContextsPlugin();
});
}
var runtime = module.exports = {
init: init,
start: start,
2014-11-04 12:34:49 +01:00
stop: stop,
2014-11-21 17:35:29 +01:00
version: getVersion,
get log() { return log },
get i18n() { return i18n },
settings: settings,
storage: storage,
events: events,
2015-11-24 23:38:42 +01:00
nodes: redNodes,
library: library,
2015-11-24 23:38:42 +01:00
util: require("./util"),
get adminApi() { return adminApi },
get nodeApp() { return nodeApp },
isStarted: function() {
return started;
}
}