1
0
mirror of https://github.com/node-red/node-red.git synced 2023-10-10 13:36:53 +02:00
node-red/nodes/core/io/32-udp.js

262 lines
10 KiB
JavaScript
Raw Permalink Normal View History

/**
* Copyright JS Foundation and other contributors, http://js.foundation
*
* 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.
**/
2014-05-04 00:32:04 +02:00
module.exports = function(RED) {
"use strict";
var os = require('os');
2014-05-04 00:32:04 +02:00
var dgram = require('dgram');
var udpInputPortsInUse = {};
2014-05-04 00:32:04 +02:00
// The Input Node
function UDPin(n) {
RED.nodes.createNode(this,n);
this.group = n.group;
this.port = n.port;
this.datatype = n.datatype;
this.iface = n.iface || null;
this.multicast = n.multicast;
2015-03-05 14:07:38 +01:00
this.ipv = n.ipv || "udp4";
2014-05-04 00:32:04 +02:00
var node = this;
if (node.iface && node.iface.indexOf(".") === -1) {
try {
if ((os.networkInterfaces())[node.iface][0].hasOwnProperty("scopeid")) {
if (node.ipv === "udp4") {
node.iface = (os.networkInterfaces())[node.iface][1].address;
} else {
node.iface = (os.networkInterfaces())[node.iface][0].address;
}
}
else {
if (node.ipv === "udp4") {
node.iface = (os.networkInterfaces())[node.iface][0].address;
} else {
node.iface = (os.networkInterfaces())[node.iface][1].address;
}
}
}
catch(e) {
node.warn(RED._("udp.errors.ifnotfound",{iface:node.iface}));
node.iface = null;
}
}
var opts = {type:node.ipv, reuseAddr:true};
if (process.version.indexOf("v0.10") === 0) { opts = node.ipv; }
var server;
if (!udpInputPortsInUse.hasOwnProperty(this.port)) {
server = dgram.createSocket(opts); // default to udp4
udpInputPortsInUse[this.port] = server;
}
else {
node.log(RED._("udp.errors.alreadyused",{port:node.port}));
server = udpInputPortsInUse[this.port]; // re-use existing
}
if (process.version.indexOf("v0.10") === 0) { opts = node.ipv; }
2014-05-04 00:32:04 +02:00
server.on("error", function (err) {
if ((err.code == "EACCES") && (node.port < 1024)) {
node.error(RED._("udp.errors.access-error"));
} else {
2015-05-28 22:55:22 +02:00
node.error(RED._("udp.errors.error",{error:err.code}));
}
server.close();
2014-05-04 00:32:04 +02:00
});
2014-05-04 00:32:04 +02:00
server.on('message', function (message, remote) {
var msg;
if (node.datatype =="base64") {
msg = { payload:message.toString('base64'), fromip:remote.address+':'+remote.port, ip:remote.address, port:remote.port };
2014-05-04 00:32:04 +02:00
} else if (node.datatype =="utf8") {
msg = { payload:message.toString('utf8'), fromip:remote.address+':'+remote.port, ip:remote.address, port:remote.port };
2014-05-04 00:32:04 +02:00
} else {
msg = { payload:message, fromip:remote.address+':'+remote.port, ip:remote.address, port:remote.port };
}
node.send(msg);
});
2014-05-04 00:32:04 +02:00
server.on('listening', function () {
var address = server.address();
2015-05-28 22:55:22 +02:00
node.log(RED._("udp.status.listener-at",{host:address.address,port:address.port}));
2014-05-04 00:32:04 +02:00
if (node.multicast == "true") {
server.setBroadcast(true);
try {
2014-05-04 00:32:04 +02:00
server.setMulticastTTL(128);
server.addMembership(node.group,node.iface);
2015-05-28 22:55:22 +02:00
node.log(RED._("udp.status.mc-group",{group:node.group}));
} catch (e) {
if (e.errno == "EINVAL") {
node.error(RED._("udp.errors.bad-mcaddress"));
2014-03-31 00:05:59 +02:00
} else if (e.errno == "ENODEV") {
node.error(RED._("udp.errors.interface"));
2014-03-31 00:05:59 +02:00
} else {
2015-05-28 22:55:22 +02:00
node.error(RED._("udp.errors.error",{error:e.errno}));
2014-03-31 00:05:59 +02:00
}
}
}
});
2014-05-04 00:32:04 +02:00
node.on("close", function() {
if (udpInputPortsInUse.hasOwnProperty(node.port)) {
delete udpInputPortsInUse[node.port];
}
2014-05-04 00:32:04 +02:00
try {
server.close();
2015-05-28 22:55:22 +02:00
node.log(RED._("udp.status.listener-stopped"));
2014-05-04 00:32:04 +02:00
} catch (err) {
//node.error(err);
2014-05-04 00:32:04 +02:00
}
});
try { server.bind(node.port,node.iface); }
catch(e) { } // Don't worry if already bound
}
RED.httpAdmin.get('/udp-ports/:id', RED.auth.needsPermission('udp-ports.read'), function(req,res) {
res.json(Object.keys(udpInputPortsInUse));
});
2014-05-04 00:32:04 +02:00
RED.nodes.registerType("udp in",UDPin);
2014-05-04 00:32:04 +02:00
// The Output Node
function UDPout(n) {
RED.nodes.createNode(this,n);
//this.group = n.group;
this.port = n.port;
this.outport = n.outport||"";
this.base64 = n.base64;
this.addr = n.addr;
this.iface = n.iface || null;
this.multicast = n.multicast;
2015-03-05 14:07:38 +01:00
this.ipv = n.ipv || "udp4";
2014-05-04 00:32:04 +02:00
var node = this;
if (node.iface && node.iface.indexOf(".") === -1) {
try {
if ((os.networkInterfaces())[node.iface][0].hasOwnProperty("scopeid")) {
if (node.ipv === "udp4") {
node.iface = (os.networkInterfaces())[node.iface][1].address;
} else {
node.iface = (os.networkInterfaces())[node.iface][0].address;
}
}
else {
if (node.ipv === "udp4") {
node.iface = (os.networkInterfaces())[node.iface][0].address;
} else {
node.iface = (os.networkInterfaces())[node.iface][1].address;
}
}
}
catch(e) {
node.warn(RED._("udp.errors.ifnotfound",{iface:node.iface}));
node.iface = null;
}
}
var opts = {type:node.ipv, reuseAddr:true};
if (process.version.indexOf("v0.10") === 0) { opts = node.ipv; }
var sock;
var p = this.outport || this.port || "0";
if (udpInputPortsInUse[p]) {
sock = udpInputPortsInUse[p];
node.log(RED._("udp.status.re-use",{outport:node.outport,host:node.addr,port:node.port}));
}
else {
sock = dgram.createSocket(opts); // default to udp4
sock.on("error", function(err) {
// Any async error will also get reported in the sock.send call.
// This handler is needed to ensure the error marked as handled to
// prevent it going to the global error handler and shutting node-red
// down.
});
udpInputPortsInUse[p] = sock;
if (node.multicast != "false") {
sock.bind(node.outport, function() { // have to bind before you can enable broadcast...
sock.setBroadcast(true); // turn on broadcast
if (node.multicast == "multi") {
try {
sock.setMulticastTTL(128);
sock.addMembership(node.addr,node.iface); // Add to the multicast group
node.log(RED._("udp.status.mc-ready",{iface:node.iface,outport:node.outport,host:node.addr,port:node.port}));
} catch (e) {
if (e.errno == "EINVAL") {
node.error(RED._("udp.errors.bad-mcaddress"));
} else if (e.errno == "ENODEV") {
node.error(RED._("udp.errors.interface"));
} else {
node.error(RED._("udp.errors.error",{error:e.errno}));
}
2014-05-04 00:32:04 +02:00
}
} else {
node.log(RED._("udp.status.bc-ready",{outport:node.outport,host:node.addr,port:node.port}));
2014-05-04 00:32:04 +02:00
}
});
} else if ((node.outport !== "") && (!udpInputPortsInUse[node.outport])) {
sock.bind(node.outport);
node.log(RED._("udp.status.ready",{outport:node.outport,host:node.addr,port:node.port}));
} else {
node.log(RED._("udp.status.ready-nolocal",{host:node.addr,port:node.port}));
}
2014-05-04 00:32:04 +02:00
}
2014-05-04 00:32:04 +02:00
node.on("input", function(msg) {
if (msg.hasOwnProperty("payload")) {
2014-05-04 00:32:04 +02:00
var add = node.addr || msg.ip || "";
var por = node.port || msg.port || 0;
if (add === "") {
node.warn(RED._("udp.errors.ip-notset"));
} else if (por === 0) {
node.warn(RED._("udp.errors.port-notset"));
2014-05-04 00:32:04 +02:00
} else if (isNaN(por) || (por < 1) || (por > 65535)) {
node.warn(RED._("udp.errors.port-invalid"));
2014-05-04 00:32:04 +02:00
} else {
var message;
if (node.base64) {
message = Buffer.from(msg.payload, 'base64');
2014-05-04 00:32:04 +02:00
} else if (msg.payload instanceof Buffer) {
message = msg.payload;
} else {
message = Buffer.from(""+msg.payload);
2014-03-31 00:05:59 +02:00
}
2014-05-04 00:32:04 +02:00
sock.send(message, 0, message.length, por, add, function(err, bytes) {
if (err) {
2015-03-16 14:58:01 +01:00
node.error("udp : "+err,msg);
2014-05-04 00:32:04 +02:00
}
message = null;
2014-05-04 00:32:04 +02:00
});
}
}
2014-05-04 00:32:04 +02:00
});
2014-05-04 00:32:04 +02:00
node.on("close", function() {
if (udpInputPortsInUse.hasOwnProperty(p)) {
delete udpInputPortsInUse[p];
}
2014-05-04 00:32:04 +02:00
try {
sock.close();
2015-05-28 22:55:22 +02:00
node.log(RED._("udp.status.output-stopped"));
2014-05-04 00:32:04 +02:00
} catch (err) {
//node.error(err);
2014-05-04 00:32:04 +02:00
}
});
}
RED.nodes.registerType("udp out",UDPout);
}