1
0
mirror of https://github.com/node-red/node-red.git synced 2023-10-10 13:36:53 +02:00

Merge branch 'master' into dev

This commit is contained in:
Nick O'Leary 2018-09-06 10:28:07 +01:00
commit e939d5e96e
No known key found for this signature in database
GPG Key ID: 4F2157149161A6C9
10 changed files with 136 additions and 59 deletions

View File

@ -1,3 +1,19 @@
#### 0.19.3: Maintenance Release
- Split node - fix complete to send msg for k/v object
- Remove unused Join node merged object key typed input
- Set the JavaScript editor to full-screen
- Filter global modules installed locally
- Add svg to permitted icon extension list
- Debug node - indicate status all the time if selected to do so
- pi nodes - increase test coverage slightly
- TCP-request node - only write payload
- JSON schema: perform validation when obj -> obj or str -> str
- JSON schema: add draft-06 support (via $schema keyword)
- Mqtt proxy configuration for websocket connection, #1651.
- Allows MQTT Shared Subscriptions for MQTT-In core node
- Fix use of HTML tag or CSS class specification as icon of typedInput
#### 0.19.2: Maintenance Release #### 0.19.2: Maintenance Release
- Ensure node default colour is used if palette.theme has no match - Ensure node default colour is used if palette.theme has no match

View File

@ -32,7 +32,7 @@ RED.editor.types._js = (function() {
var trayOptions = { var trayOptions = {
title: options.title, title: options.title,
width: "inherit", width: options.width||"inherit",
buttons: [ buttons: [
{ {
id: "node-dialog-cancel", id: "node-dialog-cancel",

View File

@ -20,7 +20,7 @@ module.exports = function(RED) {
this.severity = n.severity || 40; this.severity = n.severity || 40;
this.active = (n.active === null || typeof n.active === "undefined") || n.active; this.active = (n.active === null || typeof n.active === "undefined") || n.active;
if (this.tostatus) { if (this.tostatus) {
this.oldStatus = {fill:"grey", shape:this.active?"dot":"ring"}; this.oldStatus = {fill:"grey", shape:"ring"};
this.status(this.oldStatus); this.status(this.oldStatus);
} }
else { this.status({}); } else { this.status({}); }
@ -131,7 +131,7 @@ module.exports = function(RED) {
node.active = false; node.active = false;
res.sendStatus(201); res.sendStatus(201);
if (node.tostatus && node.hasOwnProperty("oldStatus")) { if (node.tostatus && node.hasOwnProperty("oldStatus")) {
node.oldStatus.shape = "ring"; node.oldStatus.shape = "dot";
node.status(node.oldStatus); node.status(node.oldStatus);
} }
} else { } else {

View File

@ -128,6 +128,7 @@
var value = that.editor.getValue(); var value = that.editor.getValue();
RED.editor.editJavaScript({ RED.editor.editJavaScript({
value: value, value: value,
width: "Infinity",
cursor: that.editor.getCursorPosition(), cursor: that.editor.getCursorPosition(),
complete: function(v,cursor) { complete: function(v,cursor) {
that.editor.setValue(v, -1); that.editor.setValue(v, -1);

View File

@ -295,7 +295,8 @@
For object outputs, once this count has been reached, the node can be configured to send a message for each subsequent message For object outputs, once this count has been reached, the node can be configured to send a message for each subsequent message
received.</p> received.</p>
<p>A <i>timeout</i> can be set to trigger sending the new message using whatever has been received so far.</p> <p>A <i>timeout</i> can be set to trigger sending the new message using whatever has been received so far.</p>
<p>If a message is received with the <b>msg.complete</b> property set, the output message is sent.</p> <p>If a message is received with the <b>msg.complete</b> property set, the output message is finalised and sent.
This resets any part counts.</p>
<h4>Reduce Sequence mode</h4> <h4>Reduce Sequence mode</h4>
<p>When configured to join in reduce mode, an expression is applied to each <p>When configured to join in reduce mode, an expression is applied to each
@ -439,10 +440,7 @@
$("#node-input-joiner").typedInput({ $("#node-input-joiner").typedInput({
default: 'str', default: 'str',
typeField: $("#node-input-joinerType"), typeField: $("#node-input-joinerType"),
types:[ types:['str', 'bin']
'str',
'bin'
]
}); });
$("#node-input-property").typedInput({ $("#node-input-property").typedInput({
@ -451,7 +449,7 @@
}); });
$("#node-input-key").typedInput({ $("#node-input-key").typedInput({
types:['msg', {value:"merge", label:"", hasValue:false}] types:['msg']
}); });
$("#node-input-build").change(); $("#node-input-build").change();

View File

@ -586,14 +586,18 @@ module.exports = function(RED) {
} }
else { else {
if (msg.hasOwnProperty('complete')) { if (msg.hasOwnProperty('complete')) {
if (inflight[partId]) {
inflight[partId].msg.complete = msg.complete;
completeSend(partId); completeSend(partId);
} }
}
else { else {
node.warn("Message missing key property 'msg."+node.key+"' - cannot add to object") node.warn("Message missing key property 'msg."+node.key+"' - cannot add to object")
} }
} }
return; return;
} }
if (!inflight.hasOwnProperty(partId)) { if (!inflight.hasOwnProperty(partId)) {
if (payloadType === 'object' || payloadType === 'merged') { if (payloadType === 'object' || payloadType === 'merged') {
inflight[partId] = { inflight[partId] = {
@ -604,19 +608,6 @@ module.exports = function(RED) {
msg:RED.util.cloneMessage(msg) msg:RED.util.cloneMessage(msg)
}; };
} }
else if (node.accumulate === true) {
if (msg.hasOwnProperty("reset")) { delete inflight[partId]; }
inflight[partId] = inflight[partId] || {
currentCount:0,
payload:{},
targetCount:targetCount,
type:payloadType,
msg:RED.util.cloneMessage(msg)
}
if (payloadType === 'string' || payloadType === 'array' || payloadType === 'buffer') {
inflight[partId].payload = [];
}
}
else { else {
inflight[partId] = { inflight[partId] = {
currentCount:0, currentCount:0,

View File

@ -196,9 +196,10 @@ module.exports = function(RED) {
}).catch(err => { }).catch(err => {
node.error(err,msg); node.error(err,msg);
}); });
return;
} }
var parts = msg.parts; var parts = msg.parts;
if (!parts.hasOwnProperty("id") || !parts.hasOwnProperty("index")) { if (!parts || !parts.hasOwnProperty("id") || !parts.hasOwnProperty("index")) {
return; return;
} }
var gid = parts.id; var gid = parts.id;
@ -242,7 +243,8 @@ module.exports = function(RED) {
delete pending[key]; delete pending[key];
} }
} }
pending_count = 0; }) pending_count = 0;
});
} }
RED.nodes.registerType("sort", SortNode); RED.nodes.registerType("sort", SortNode);

View File

@ -1,6 +1,6 @@
{ {
"name": "node-red", "name": "node-red",
"version": "0.19.2", "version": "0.19.3",
"description": "A visual tool for wiring the Internet of Things", "description": "A visual tool for wiring the Internet of Things",
"homepage": "http://nodered.org", "homepage": "http://nodered.org",
"license": "Apache-2.0", "license": "Apache-2.0",
@ -42,7 +42,7 @@
"cookie": "0.3.1", "cookie": "0.3.1",
"cookie-parser": "1.4.3", "cookie-parser": "1.4.3",
"cors": "2.8.4", "cors": "2.8.4",
"cron": "1.3.0", "cron": "1.4.1",
"denque": "1.3.0", "denque": "1.3.0",
"express": "4.16.3", "express": "4.16.3",
"express-session": "1.15.6", "express-session": "1.15.6",
@ -57,9 +57,9 @@
"jsonata": "1.5.4", "jsonata": "1.5.4",
"media-typer": "0.3.0", "media-typer": "0.3.0",
"memorystore": "1.6.0", "memorystore": "1.6.0",
"mqtt": "2.18.5", "mqtt": "2.18.8",
"multer": "1.3.1", "multer": "1.3.1",
"mustache": "2.3.1", "mustache": "2.3.2",
"node-red-node-email": "0.1.*", "node-red-node-email": "0.1.*",
"node-red-node-feedparser": "^0.1.12", "node-red-node-feedparser": "^0.1.12",
"node-red-node-rbe": "0.2.*", "node-red-node-rbe": "0.2.*",
@ -74,7 +74,7 @@
"request": "2.88.0", "request": "2.88.0",
"semver": "5.5.1", "semver": "5.5.1",
"sentiment": "2.1.0", "sentiment": "2.1.0",
"uglify-js": "3.4.8", "uglify-js": "3.4.9",
"when": "3.7.8", "when": "3.7.8",
"ws": "1.1.5", "ws": "1.1.5",
"xml2js": "0.4.19" "xml2js": "0.4.19"
@ -86,7 +86,7 @@
"chromedriver": "^2.41.0", "chromedriver": "^2.41.0",
"grunt": "~1.0.3", "grunt": "~1.0.3",
"grunt-chmod": "~1.1.1", "grunt-chmod": "~1.1.1",
"grunt-cli": "~1.2.0", "grunt-cli": "~1.3.1",
"grunt-concurrent": "~2.3.1", "grunt-concurrent": "~2.3.1",
"grunt-contrib-clean": "~1.1.0", "grunt-contrib-clean": "~1.1.0",
"grunt-contrib-compress": "~1.4.0", "grunt-contrib-compress": "~1.4.0",

View File

@ -603,14 +603,14 @@ describe('JOIN node', function() {
}); });
it('should accumulate a merged object', function(done) { it('should accumulate a merged object', function(done) {
var flow = [{id:"n1", type:"join", wires:[["n2"]], build:"merged",mode:"custom",accumulate:true, count:1}, var flow = [{id:"n1", type:"join", wires:[["n2"]], build:"merged",mode:"custom",accumulate:true, count:3},
{id:"n2", type:"helper"}]; {id:"n2", type:"helper"}];
helper.load(joinNode, flow, function() { helper.load(joinNode, flow, function() {
var n1 = helper.getNode("n1"); var n1 = helper.getNode("n1");
var n2 = helper.getNode("n2"); var n2 = helper.getNode("n2");
var c = 0; var c = 0;
n2.on("input", function(msg) { n2.on("input", function(msg) {
if (c === 5) { if (c === 3) {
try { try {
msg.should.have.property("payload"); msg.should.have.property("payload");
msg.payload.should.have.property("a",3); msg.payload.should.have.property("a",3);
@ -632,14 +632,14 @@ describe('JOIN node', function() {
}); });
it('should be able to reset an accumulation', function(done) { it('should be able to reset an accumulation', function(done) {
var flow = [{id:"n1", type:"join", wires:[["n2"]], build:"merged",accumulate:true,mode:"custom", count:1}, var flow = [{id:"n1", type:"join", wires:[["n2"]], build:"merged",accumulate:true,mode:"custom", count:3},
{id:"n2", type:"helper"}]; {id:"n2", type:"helper"}];
helper.load(joinNode, flow, function() { helper.load(joinNode, flow, function() {
var n1 = helper.getNode("n1"); var n1 = helper.getNode("n1");
var n2 = helper.getNode("n2"); var n2 = helper.getNode("n2");
var c = 0; var c = 0;
n2.on("input", function(msg) { n2.on("input", function(msg) {
if (c === 3) { if (c === 1) {
try { try {
msg.should.have.property("payload"); msg.should.have.property("payload");
msg.payload.should.have.property("a",1); msg.payload.should.have.property("a",1);
@ -649,11 +649,20 @@ describe('JOIN node', function() {
} }
catch(e) { done(e) } catch(e) { done(e) }
} }
if (c === 5) { if (c === 2) {
try { try {
msg.should.have.property("payload"); msg.should.have.property("payload");
msg.payload.should.have.property("b",2); msg.payload.should.have.property("e",2);
msg.payload.should.have.property("c",1); msg.payload.should.have.property("f",1);
}
catch(e) { done(e) }
}
if (c === 3) {
try {
msg.should.have.property("payload");
msg.payload.should.have.property("g",2);
msg.payload.should.have.property("h",1);
msg.payload.should.have.property("i",3);
done(); done();
} }
catch(e) { done(e) } catch(e) { done(e) }
@ -664,8 +673,11 @@ describe('JOIN node', function() {
n1.receive({payload:{b:2}, topic:"b"}); n1.receive({payload:{b:2}, topic:"b"});
n1.receive({payload:{c:3}, topic:"c"}); n1.receive({payload:{c:3}, topic:"c"});
n1.receive({payload:{d:4}, topic:"d", complete:true}); n1.receive({payload:{d:4}, topic:"d", complete:true});
n1.receive({payload:{b:2}, topic:"e"}); n1.receive({payload:{e:2}, topic:"e"});
n1.receive({payload:{c:1}, topic:"f"}); n1.receive({payload:{f:1}, topic:"f", complete:true});
n1.receive({payload:{g:2}, topic:"g"});
n1.receive({payload:{h:1}, topic:"h"});
n1.receive({payload:{i:3}, topic:"i"});
}); });
}); });

View File

@ -66,13 +66,25 @@ describe('SORT node', function() {
var n1 = helper.getNode("n1"); var n1 = helper.getNode("n1");
var n2 = helper.getNode("n2"); var n2 = helper.getNode("n2");
n2.on("input", function(msg) { n2.on("input", function(msg) {
try {
msg.should.have.property(target); msg.should.have.property(target);
var data = msg[target]; var data = msg[target];
data.length.should.equal(data_out.length); data.length.should.equal(data_out.length);
for(var i = 0; i < data_out.length; i++) { for(var i = 0; i < data_out.length; i++) {
data[i].should.equal(data_out[i]); var data0 = data[i];
var data1 = data_out[i];
if (typeof data0 === "object") {
data0.should.deepEqual(data1);
}
else {
data0.should.equal(data1);
}
} }
done(); done();
}
catch(e) {
console.log(e);
}
}); });
var msg = {}; var msg = {};
msg[target] = data_in; msg[target] = data_in;
@ -93,6 +105,34 @@ describe('SORT node', function() {
} }
function check_sort1(flow, key, key_type, data_in, data_out, done) { function check_sort1(flow, key, key_type, data_in, data_out, done) {
function equals(v0, v1) {
var k0 = Object.keys(v0);
var k1 = Object.keys(v1);
if (k0.length === k1.length) {
for (var i = 0; i < k0.length; i++) {
var k = k0[i];
if (!v1.hasOwnProperty(k) ||
(v0[k] !== v1[k])) {
return false;
}
}
return true;
}
return false;
}
function indexOf(a, v) {
for(var i = 0; i < a.length; i++) {
var av = a[i];
if ((typeof v === 'object') && equals(v, av)) {
return i;
}
else if (v === av) {
return i;
}
}
return -1;
}
var sort = flow[0]; var sort = flow[0];
var prop = (key_type === "msg") ? key : "payload"; var prop = (key_type === "msg") ? key : "payload";
sort.targetType = "seq"; sort.targetType = "seq";
@ -107,7 +147,7 @@ describe('SORT node', function() {
msg.should.have.property("parts"); msg.should.have.property("parts");
msg.parts.should.have.property("count", data_out.length); msg.parts.should.have.property("count", data_out.length);
var data = msg[prop]; var data = msg[prop];
var index = data_out.indexOf(data); var index = indexOf(data_out, data);
msg.parts.should.have.property("index", index); msg.parts.should.have.property("index", index);
count++; count++;
if (count === data_out.length) { if (count === data_out.length) {
@ -136,7 +176,6 @@ describe('SORT node', function() {
check_sort1(flow, exp, "jsonata", data_in, data_out, done); check_sort1(flow, exp, "jsonata", data_in, data_out, done);
} }
(function() { (function() {
var flow = [{id:"n1", type:"sort", order:"ascending", as_num:false, wires:[["n2"]]}, var flow = [{id:"n1", type:"sort", order:"ascending", as_num:false, wires:[["n2"]]},
{id:"n2", type:"helper"}]; {id:"n2", type:"helper"}];
@ -239,6 +278,19 @@ describe('SORT node', function() {
}); });
})(); })();
(function() {
var flow = [{id:"n1", type:"sort", order:"ascending", as_num:true, wires:[["n2"]]},
{id:"n2", type:"helper"}];
var conv = function(x) {
return x.map(function(v) { return { val:v }; });
};
var data_in = conv([ "200", "4", "30", "1000" ]);
var data_out = conv([ "4", "30", "200", "1000" ]);
it('should sort payload of objects', function(done) {
check_sort0C(flow, "val", data_in, data_out, done);
});
})();
it('should sort payload by context (exp, not number, ascending)', function(done) { it('should sort payload by context (exp, not number, ascending)', function(done) {
var flow = [{id:"n1", type:"sort", target:"data", targetType:"msg", msgKey:"$flowContext($)", msgKeyType:"jsonata", order:"ascending", as_num:false, wires:[["n2"]],z:"flow"}, var flow = [{id:"n1", type:"sort", target:"data", targetType:"msg", msgKey:"$flowContext($)", msgKeyType:"jsonata", order:"ascending", as_num:false, wires:[["n2"]],z:"flow"},
{id:"n2", type:"helper",z:"flow"}, {id:"n2", type:"helper",z:"flow"},
@ -268,7 +320,7 @@ describe('SORT node', function() {
}); });
it('should sort message group by context (exp, not number, ascending)', function(done) { it('should sort message group by context (exp, not number, ascending)', function(done) {
var flow = [{id:"n1", type:"sort", target:"data", targetType:"msg", msgKey:"$globalContext(payload)", msgKeyType:"jsonata", order:"ascending", as_num:false, wires:[["n2"]],z:"flow"}, var flow = [{id:"n1", type:"sort", target:"data", targetType:"seq", seqKey:"$globalContext(payload)", seqKeyType:"jsonata", order:"ascending", as_num:false, wires:[["n2"]],z:"flow"},
{id:"n2", type:"helper",z:"flow"}, {id:"n2", type:"helper",z:"flow"},
{id:"flow", type:"tab"}]; {id:"flow", type:"tab"}];
var data_in = [ "first", "second", "third", "fourth" ]; var data_in = [ "first", "second", "third", "fourth" ];
@ -282,6 +334,7 @@ describe('SORT node', function() {
n1.context()["global"].set("third","3"); n1.context()["global"].set("third","3");
n1.context()["global"].set("fourth","2"); n1.context()["global"].set("fourth","2");
n2.on("input", function(msg) { n2.on("input", function(msg) {
try {
msg.should.have.property("payload"); msg.should.have.property("payload");
msg.should.have.property("parts"); msg.should.have.property("parts");
msg.parts.should.have.property("count", data_out.length); msg.parts.should.have.property("count", data_out.length);
@ -292,6 +345,10 @@ describe('SORT node', function() {
if (count === data_out.length) { if (count === data_out.length) {
done(); done();
} }
}
catch(e) {
done(e);
}
}); });
var len = data_in.length; var len = data_in.length;
for(var i = 0; i < len; i++) { for(var i = 0; i < len; i++) {
@ -332,7 +389,7 @@ describe('SORT node', function() {
}); });
it('should sort message group by persistable context (exp, not number, descending)', function(done) { it('should sort message group by persistable context (exp, not number, descending)', function(done) {
var flow = [{id:"n1", type:"sort", target:"data", targetType:"msg", msgKey:"$flowContext(payload,\"memory\")", msgKeyType:"jsonata", order:"descending", as_num:false, wires:[["n2"]],z:"flow"}, var flow = [{id:"n1", type:"sort", target:"data", targetType:"seq", seqKey:"$flowContext(payload,\"memory\")", seqKeyType:"jsonata", order:"descending", as_num:false, wires:[["n2"]],z:"flow"},
{id:"n2", type:"helper",z:"flow"}, {id:"n2", type:"helper",z:"flow"},
{id:"flow", type:"tab"}]; {id:"flow", type:"tab"}];
var data_in = [ "first", "second", "third", "fourth" ]; var data_in = [ "first", "second", "third", "fourth" ];