/** * Copyright 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 fs = require("fs"); var path = require("path"); var events = require("../../events"); var localfilesystem = require("./localfilesystem"); var registry = require("./registry"); var RED; var settings; var i18n = require("../../i18n"); events.on("node-locales-dir", function(info) { i18n.registerMessageCatalog(info.namespace,info.dir,info.file); }); function init(_settings) { settings = _settings; localfilesystem.init(settings); RED = require('../../red'); } function load(defaultNodesDir,disableNodePathScan) { // To skip node scan, the following line will use the stored node list. // We should expose that as an option at some point, although the // performance gains are minimal. //return loadNodeFiles(registry.getModuleList()); var nodeFiles = localfilesystem.getNodeFiles(defaultNodesDir,disableNodePathScan); return loadNodeFiles(nodeFiles); } function loadNodeFiles(nodeFiles) { var promises = []; for (var module in nodeFiles) { /* istanbul ignore else */ if (nodeFiles.hasOwnProperty(module)) { if (module == "node-red" || !registry.getModuleInfo(module)) { var first = true; for (var node in nodeFiles[module].nodes) { /* istanbul ignore else */ if (nodeFiles[module].nodes.hasOwnProperty(node)) { if (module != "node-red" && first) { // Check the module directory exists first = false; var fn = nodeFiles[module].nodes[node].file; var parts = fn.split("/"); var i = parts.length-1; for (;i>=0;i--) { if (parts[i] == "node_modules") { break; } } var moduleFn = parts.slice(0,i+2).join("/"); try { var stat = fs.statSync(moduleFn); } catch(err) { // Module not found, don't attempt to load its nodes break; } } try { promises.push(loadNodeConfig(nodeFiles[module].nodes[node])) } catch(err) { // } } } } } } return when.settle(promises).then(function(results) { var nodes = results.map(function(r) { registry.addNodeSet(r.value.id,r.value,r.value.version); return r.value; }); return loadNodeSetList(nodes); }); } function loadNodeConfig(fileInfo) { return when.promise(function(resolve) { var file = fileInfo.file; var module = fileInfo.module; var name = fileInfo.name; var version = fileInfo.version; var id = module + "/" + name; var info = registry.getNodeInfo(id); var isEnabled = true; if (info) { if (info.hasOwnProperty("loaded")) { throw new Error(file+" already loaded"); } isEnabled = info.enabled; } var node = { id: id, module: module, name: name, file: file, template: file.replace(/\.js$/,".html"), enabled: isEnabled, loaded:false, version: version }; if (fileInfo.hasOwnProperty("types")) { node.types = fileInfo.types; } fs.readFile(node.template,'utf8', function(err,content) { if (err) { node.types = []; if (err.code === 'ENOENT') { if (!node.types) { node.types = []; } node.err = "Error: "+file+" does not exist"; } else { node.types = []; node.err = err.toString(); } resolve(node); } else { var types = []; var regExp = /