mirror of
https://github.com/node-red/node-red.git
synced 2023-10-10 13:36:53 +02:00
94 lines
3.6 KiB
JavaScript
94 lines
3.6 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.
|
|
**/
|
|
|
|
module.exports = function(RED) {
|
|
"use strict";
|
|
var util = require("util");
|
|
var vm = require("vm");
|
|
|
|
function FunctionNode(n) {
|
|
RED.nodes.createNode(this,n);
|
|
this.name = n.name;
|
|
this.func = n.func;
|
|
var functionText = "var results = null; results = (function(msg){\n"+this.func+"\n})(msg);";
|
|
this.topic = n.topic;
|
|
var sandbox = {
|
|
console:console,
|
|
util:util,
|
|
Buffer:Buffer,
|
|
context: {
|
|
global:RED.settings.functionGlobalContext || {}
|
|
}
|
|
};
|
|
var context = vm.createContext(sandbox);
|
|
try {
|
|
this.script = vm.createScript(functionText);
|
|
this.on("input", function(msg) {
|
|
try {
|
|
var start = process.hrtime();
|
|
context.msg = msg;
|
|
this.script.runInContext(context);
|
|
var results = context.results;
|
|
if (results == null) {
|
|
results = [];
|
|
} else if (results.length == null) {
|
|
results = [results];
|
|
}
|
|
if (msg._topic) {
|
|
for (var m in results) {
|
|
if (results[m]) {
|
|
if (util.isArray(results[m])) {
|
|
for (var n=0; n < results[m].length; n++) {
|
|
results[m][n]._topic = msg._topic;
|
|
}
|
|
} else {
|
|
results[m]._topic = msg._topic;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
this.send(results);
|
|
var duration = process.hrtime(start);
|
|
var converted = Math.floor((duration[0]* 1e9 + duration[1])/10000)/100;
|
|
this.metric("duration", msg, converted);
|
|
if (process.env.NODE_RED_FUNCTION_TIME) {
|
|
this.status({fill:"yellow",shape:"dot",text:""+converted});
|
|
}
|
|
} catch(err) {
|
|
var errorMessage = err.toString();
|
|
var stack = err.stack.split(/\r?\n/);
|
|
if (stack.length > 0) {
|
|
var m = /at undefined:(\d+):(\d+)$/.exec(stack[1]);
|
|
if (m) {
|
|
var line = Number(m[1])-1;
|
|
var cha = m[2];
|
|
errorMessage += " (line "+line+", col "+cha+")";
|
|
}
|
|
}
|
|
this.error(errorMessage);
|
|
}
|
|
});
|
|
} catch(err) {
|
|
// eg SyntaxError - which v8 doesn't include line number information
|
|
// so we can't do better than this
|
|
this.error(err);
|
|
}
|
|
}
|
|
|
|
RED.nodes.registerType("function",FunctionNode);
|
|
RED.library.register("functions");
|
|
}
|