2016-06-04 00:40:40 +01:00
|
|
|
/**
|
|
|
|
* Copyright 2016 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";
|
|
|
|
|
|
|
|
function SplitNode(n) {
|
|
|
|
RED.nodes.createNode(this,n);
|
2016-06-09 11:33:40 +01:00
|
|
|
this.splt = (n.splt || "\\n").replace(/\\n/,"\n").replace(/\\r/,"\r").replace(/\\t/,"\t").replace(/\\e/,"\e").replace(/\\f/,"\f").replace(/\\0/,"\0");
|
2016-06-04 00:40:40 +01:00
|
|
|
var node = this;
|
|
|
|
this.on("input", function(msg) {
|
|
|
|
if (msg.hasOwnProperty("payload")) {
|
|
|
|
var a = msg.payload;
|
|
|
|
if (msg.hasOwnProperty("parts")) { msg.parts = { parts:msg.parts }; } // push existing parts to a stack
|
|
|
|
else { msg.parts = {}; }
|
|
|
|
msg.parts.id = msg._msgid; // use the existing _msgid by default.
|
|
|
|
if (typeof msg.payload === "string") { // Split String into array
|
2016-06-09 11:33:40 +01:00
|
|
|
a = msg.payload.split(node.splt);
|
2016-06-04 00:40:40 +01:00
|
|
|
msg.parts.ch = node.splt; // pass the split char to other end for rejoin
|
|
|
|
msg.parts.type = "string";
|
|
|
|
}
|
|
|
|
if (Array.isArray(a)) { // then split array into messages
|
|
|
|
msg.parts.type = msg.parts.type || "array"; // if it wasn't a string in the first place
|
|
|
|
for (var i = 0; i < a.length; i++) {
|
|
|
|
msg.payload = a[i];
|
|
|
|
msg.parts.index = i;
|
|
|
|
msg.parts.count = a.length;
|
2016-06-10 22:51:57 +01:00
|
|
|
node.send(RED.util.cloneMessage(msg));
|
2016-06-04 00:40:40 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
else if ((typeof msg.payload === "object") && !Buffer.isBuffer(msg.payload)) {
|
|
|
|
var j = 0;
|
|
|
|
var l = Object.keys(msg.payload).length;
|
|
|
|
var pay = msg.payload;
|
|
|
|
msg.parts.type = "object";
|
|
|
|
for (var p in pay) {
|
|
|
|
if (pay.hasOwnProperty(p)) {
|
|
|
|
msg.payload = pay[p];
|
|
|
|
msg.parts.key = p;
|
|
|
|
msg.parts.index = j;
|
|
|
|
msg.parts.count = l;
|
2016-06-10 22:51:57 +01:00
|
|
|
node.send(RED.util.cloneMessage(msg));
|
2016-06-04 00:40:40 +01:00
|
|
|
j += 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
// TODO not handling Buffers at present...
|
|
|
|
//else { } // otherwise drop the message.
|
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
|
|
|
RED.nodes.registerType("split",SplitNode);
|
|
|
|
|
|
|
|
|
|
|
|
function JoinNode(n) {
|
|
|
|
RED.nodes.createNode(this,n);
|
2016-06-09 11:33:40 +01:00
|
|
|
this.mode = n.mode||"auto";
|
2016-06-05 23:32:03 +01:00
|
|
|
this.property = n.property||"payload";
|
|
|
|
this.propertyType = n.propertyType||"msg";
|
2016-06-10 22:51:57 +01:00
|
|
|
if (this.propertyType === 'full') {
|
|
|
|
this.property = "payload";
|
|
|
|
}
|
2016-06-05 23:32:03 +01:00
|
|
|
this.key = n.key||"topic";
|
2016-06-10 22:51:57 +01:00
|
|
|
this.timer = (this.mode === "auto") ? 0 : Number(n.timeout || 0)*1000;
|
2016-06-04 00:40:40 +01:00
|
|
|
this.timerr = n.timerr || "send";
|
|
|
|
this.count = Number(n.count || 0);
|
2016-06-09 11:33:40 +01:00
|
|
|
this.joiner = (n.joiner||"").replace(/\\n/g,"\n").replace(/\\r/g,"\r").replace(/\\t/g,"\t").replace(/\\e/g,"\e").replace(/\\f/g,"\f").replace(/\\0/g,"\0");
|
2016-06-04 00:40:40 +01:00
|
|
|
this.build = n.build || "array";
|
|
|
|
var node = this;
|
|
|
|
var inflight = {};
|
|
|
|
|
2016-06-09 11:33:40 +01:00
|
|
|
var completeSend = function(partId) {
|
|
|
|
var group = inflight[partId];
|
|
|
|
clearTimeout(group.timeout);
|
|
|
|
delete inflight[partId];
|
2016-06-04 00:40:40 +01:00
|
|
|
|
2016-06-09 11:33:40 +01:00
|
|
|
if (group.type === 'string') {
|
2016-06-10 22:51:57 +01:00
|
|
|
RED.util.setMessageProperty(group.msg,node.property,group.payload.join(group.joinChar));
|
2016-06-09 11:33:40 +01:00
|
|
|
} else {
|
2016-06-10 22:51:57 +01:00
|
|
|
RED.util.setMessageProperty(group.msg,node.property,group.payload);
|
2016-06-04 00:40:40 +01:00
|
|
|
}
|
2016-06-09 11:33:40 +01:00
|
|
|
if (group.msg.hasOwnProperty('parts') && group.msg.parts.hasOwnProperty('parts')) {
|
2016-06-13 17:43:22 +01:00
|
|
|
group.msg.parts = group.msg.parts.parts;
|
2016-06-09 11:33:40 +01:00
|
|
|
} else {
|
|
|
|
delete group.msg.parts;
|
2016-06-04 00:40:40 +01:00
|
|
|
}
|
2016-06-09 11:33:40 +01:00
|
|
|
node.send(group.msg);
|
2016-06-04 00:40:40 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
this.on("input", function(msg) {
|
2016-06-09 11:33:40 +01:00
|
|
|
try {
|
2016-06-05 23:32:03 +01:00
|
|
|
var property;
|
2016-06-09 11:33:40 +01:00
|
|
|
if (node.mode === 'auto' && (!msg.hasOwnProperty("parts")||!msg.parts.hasOwnProperty("id"))) {
|
2016-06-10 22:51:57 +01:00
|
|
|
node.warn("Message missing msg.parts property - cannot join in 'auto' mode")
|
2016-06-09 11:33:40 +01:00
|
|
|
return;
|
|
|
|
}
|
2016-06-05 23:32:03 +01:00
|
|
|
if (node.propertyType == "full") {
|
|
|
|
property = msg;
|
|
|
|
} else {
|
|
|
|
try {
|
|
|
|
property = RED.util.getMessageProperty(msg,node.property);
|
|
|
|
} catch(err) {
|
|
|
|
node.warn("Message property "+node.property+" not found");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
2016-06-09 11:33:40 +01:00
|
|
|
|
|
|
|
var partId;
|
|
|
|
var payloadType;
|
|
|
|
var propertyKey;
|
|
|
|
var targetCount;
|
|
|
|
var joinChar;
|
|
|
|
var propertyIndex;
|
|
|
|
if (node.mode === "auto") {
|
|
|
|
// Use msg.parts to identify all of the group information
|
|
|
|
partId = msg.parts.id;
|
|
|
|
payloadType = msg.parts.type;
|
|
|
|
targetCount = msg.parts.count;
|
|
|
|
joinChar = msg.parts.ch;
|
|
|
|
propertyKey = msg.parts.key;
|
|
|
|
propertyIndex = msg.parts.index;
|
2016-06-05 23:32:03 +01:00
|
|
|
} else {
|
2016-06-09 11:33:40 +01:00
|
|
|
// Use the node configuration to identify all of the group information
|
|
|
|
partId = "_";
|
|
|
|
payloadType = node.build;
|
|
|
|
targetCount = node.count;
|
|
|
|
joinChar = node.joiner;
|
|
|
|
if (targetCount === 0 && msg.hasOwnProperty('parts')) {
|
|
|
|
targetCount = msg.parts.count || 0;
|
|
|
|
}
|
|
|
|
if (node.build === 'object') {
|
|
|
|
propertyKey = RED.util.getMessageProperty(msg,node.key);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (payloadType === 'object' && (propertyKey === null || propertyKey === undefined || propertyKey === "")) {
|
2016-06-10 22:51:57 +01:00
|
|
|
if (node.mode === "auto") {
|
|
|
|
node.warn("Message missing 'msg.parts.key' property - cannot add to object");
|
|
|
|
} else {
|
|
|
|
node.warn("Message missing key property 'msg."+node.key+"' '- cannot add to object")
|
|
|
|
}
|
2016-06-09 11:33:40 +01:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
if (!inflight.hasOwnProperty(partId)) {
|
2016-06-11 21:43:37 +01:00
|
|
|
if (payloadType === 'object' || payloadType === 'merged') {
|
2016-06-09 11:33:40 +01:00
|
|
|
inflight[partId] = {
|
|
|
|
currentCount:0,
|
|
|
|
payload:{},
|
|
|
|
targetCount:targetCount,
|
|
|
|
type:"object",
|
|
|
|
msg:msg
|
|
|
|
};
|
2016-06-05 23:32:03 +01:00
|
|
|
} else {
|
2016-06-09 11:33:40 +01:00
|
|
|
inflight[partId] = {
|
|
|
|
currentCount:0,
|
|
|
|
payload:[],
|
|
|
|
targetCount:targetCount,
|
|
|
|
type:payloadType,
|
|
|
|
joinChar: joinChar,
|
|
|
|
msg:msg
|
|
|
|
};
|
|
|
|
if (payloadType === 'string') {
|
|
|
|
inflight[partId].joinChar = joinChar;
|
2016-06-04 00:40:40 +01:00
|
|
|
}
|
2016-06-05 23:32:03 +01:00
|
|
|
}
|
2016-06-09 11:33:40 +01:00
|
|
|
if (node.timer > 0) {
|
|
|
|
inflight[partId].timeout = setTimeout(function() {
|
|
|
|
completeSend(partId)
|
|
|
|
}, node.timer)
|
2016-06-04 00:40:40 +01:00
|
|
|
}
|
|
|
|
}
|
2016-06-09 11:33:40 +01:00
|
|
|
|
|
|
|
var group = inflight[partId];
|
|
|
|
if (payloadType === 'object') {
|
|
|
|
group.payload[propertyKey] = property;
|
|
|
|
group.currentCount = Object.keys(group.payload).length;
|
2016-06-11 21:43:37 +01:00
|
|
|
} else if (payloadType === 'merged') {
|
|
|
|
if (Array.isArray(property) || typeof property !== 'object') {
|
|
|
|
node.warn("Cannot merge non-object types");
|
|
|
|
} else {
|
|
|
|
for (propertyKey in property) {
|
|
|
|
if (property.hasOwnProperty(propertyKey)) {
|
|
|
|
group.payload[propertyKey] = property[propertyKey];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
group.currentCount++;
|
|
|
|
}
|
2016-06-09 11:33:40 +01:00
|
|
|
} else {
|
|
|
|
if (!isNaN(propertyIndex)) {
|
|
|
|
group.payload[propertyIndex] = property;
|
|
|
|
} else {
|
|
|
|
group.payload.push(property);
|
|
|
|
}
|
|
|
|
group.currentCount++;
|
|
|
|
}
|
|
|
|
// TODO: currently reuse the last received - add option to pick first received
|
|
|
|
group.msg = msg;
|
|
|
|
if (group.currentCount === group.targetCount || msg.hasOwnProperty('complete')) {
|
|
|
|
delete msg.complete;
|
|
|
|
completeSend(partId);
|
|
|
|
}
|
|
|
|
} catch(err) {
|
|
|
|
console.log(err.stack);
|
|
|
|
}
|
2016-06-04 00:40:40 +01:00
|
|
|
});
|
|
|
|
|
|
|
|
this.on("close", function() {
|
|
|
|
for (var i in inflight) {
|
2016-06-09 11:33:40 +01:00
|
|
|
if (inflight.hasOwnProperty(i)) {
|
|
|
|
clearTimeout(inflight[i].timeout);
|
|
|
|
}
|
2016-06-04 00:40:40 +01:00
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
|
|
|
RED.nodes.registerType("join",JoinNode);
|
|
|
|
}
|