2015-01-15 17:10:32 +00:00
|
|
|
/**
|
2017-01-11 15:24:33 +00:00
|
|
|
* Copyright JS Foundation and other contributors, http://js.foundation
|
2015-01-15 17:10:32 +00:00
|
|
|
*
|
|
|
|
* 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.
|
|
|
|
**/
|
2015-10-11 20:37:11 +01:00
|
|
|
|
2015-01-15 17:10:32 +00:00
|
|
|
|
|
|
|
var should = require("should");
|
|
|
|
var sinon = require('sinon');
|
|
|
|
var clone = require('clone');
|
2015-01-16 15:43:47 +00:00
|
|
|
var util = require("util");
|
|
|
|
|
2018-08-20 16:17:24 +01:00
|
|
|
var NR_TEST_UTILS = require("nr-test-utils");
|
|
|
|
|
2020-07-20 16:48:47 +01:00
|
|
|
var flowUtils = NR_TEST_UTILS.require("@node-red/runtime/lib/flows/util");
|
|
|
|
var Flow = NR_TEST_UTILS.require("@node-red/runtime/lib/flows/Flow");
|
|
|
|
var flows = NR_TEST_UTILS.require("@node-red/runtime/lib/flows");
|
2018-08-20 16:17:24 +01:00
|
|
|
var Node = NR_TEST_UTILS.require("@node-red/runtime/lib/nodes/Node");
|
2021-04-15 15:30:02 +01:00
|
|
|
var hooks = NR_TEST_UTILS.require("@node-red/util/lib/hooks");
|
2018-08-20 16:17:24 +01:00
|
|
|
var typeRegistry = NR_TEST_UTILS.require("@node-red/registry");
|
2015-11-02 15:38:16 +00:00
|
|
|
|
|
|
|
describe('Flow', function() {
|
|
|
|
var getType;
|
|
|
|
|
|
|
|
var stoppedNodes = {};
|
2021-02-19 20:44:01 +00:00
|
|
|
var stoppedOrder = [];
|
2015-11-02 15:38:16 +00:00
|
|
|
var currentNodes = {};
|
|
|
|
var rewiredNodes = {};
|
|
|
|
var createCount = 0;
|
|
|
|
|
|
|
|
beforeEach(function() {
|
|
|
|
currentNodes = {};
|
|
|
|
stoppedNodes = {};
|
2021-02-19 20:44:01 +00:00
|
|
|
stoppedOrder =[];
|
2015-11-02 15:38:16 +00:00
|
|
|
rewiredNodes = {};
|
|
|
|
createCount = 0;
|
2018-04-23 14:24:51 +01:00
|
|
|
Flow.init({settings:{},log:{
|
2019-01-16 16:27:19 +00:00
|
|
|
log: sinon.stub(), // function() { console.log("l",[...arguments].map(a => JSON.stringify(a)).join(" ")) },//
|
|
|
|
debug: sinon.stub(), // function() { console.log("d",[...arguments].map(a => JSON.stringify(a)).join(" ")) },//sinon.stub(),
|
|
|
|
trace: sinon.stub(), // function() { console.log("t",[...arguments].map(a => JSON.stringify(a)).join(" ")) },//sinon.stub(),
|
|
|
|
warn: sinon.stub(), // function() { console.log("w",[...arguments].map(a => JSON.stringify(a)).join(" ")) },//sinon.stub(),
|
|
|
|
info: sinon.stub(), // function() { console.log("i",[...arguments].map(a => JSON.stringify(a)).join(" ")) },//sinon.stub(),
|
2018-04-23 14:24:51 +01:00
|
|
|
metric: sinon.stub(),
|
|
|
|
_: function() { return "abc"}
|
|
|
|
}});
|
2015-11-02 15:38:16 +00:00
|
|
|
});
|
2015-10-11 20:37:11 +01:00
|
|
|
|
2015-11-02 15:38:16 +00:00
|
|
|
var TestNode = function(n) {
|
|
|
|
Node.call(this,n);
|
2016-04-27 10:31:19 +01:00
|
|
|
this._index = createCount++;
|
2015-11-02 15:38:16 +00:00
|
|
|
this.scope = n.scope;
|
|
|
|
var node = this;
|
|
|
|
this.foo = n.foo;
|
|
|
|
this.handled = 0;
|
|
|
|
this.stopped = false;
|
|
|
|
currentNodes[node.id] = node;
|
|
|
|
this.on('input',function(msg) {
|
2019-01-16 16:27:19 +00:00
|
|
|
// console.log(this.id,msg.payload);
|
2015-11-02 15:38:16 +00:00
|
|
|
node.handled++;
|
|
|
|
node.send(msg);
|
|
|
|
});
|
|
|
|
this.on('close',function() {
|
|
|
|
node.stopped = true;
|
|
|
|
stoppedNodes[node.id] = node;
|
2021-02-19 20:44:01 +00:00
|
|
|
stoppedOrder.push(node.id)
|
2015-11-02 15:38:16 +00:00
|
|
|
delete currentNodes[node.id];
|
|
|
|
});
|
|
|
|
this.__updateWires = this.updateWires;
|
|
|
|
this.updateWires = function(newWires) {
|
|
|
|
rewiredNodes[node.id] = node;
|
|
|
|
node.newWires = newWires;
|
2017-05-15 13:05:33 +01:00
|
|
|
node.__updateWires(newWires);
|
2015-11-02 15:38:16 +00:00
|
|
|
};
|
|
|
|
}
|
|
|
|
util.inherits(TestNode,Node);
|
|
|
|
|
2019-01-16 16:27:19 +00:00
|
|
|
var TestErrorNode = function(n) {
|
|
|
|
Node.call(this,n);
|
|
|
|
this._index = createCount++;
|
|
|
|
this.scope = n.scope;
|
|
|
|
this.name = n.name;
|
|
|
|
var node = this;
|
|
|
|
this.foo = n.foo;
|
|
|
|
this.handled = 0;
|
|
|
|
this.stopped = false;
|
|
|
|
currentNodes[node.id] = node;
|
|
|
|
this.on('input',function(msg) {
|
|
|
|
node.handled++;
|
|
|
|
node.error("test error",msg);
|
|
|
|
});
|
|
|
|
this.on('close',function() {
|
|
|
|
node.stopped = true;
|
|
|
|
stoppedNodes[node.id] = node;
|
2021-02-19 20:44:01 +00:00
|
|
|
stoppedOrder.push(node.id)
|
2019-01-16 16:27:19 +00:00
|
|
|
delete currentNodes[node.id];
|
|
|
|
});
|
|
|
|
this.__updateWires = this.updateWires;
|
|
|
|
this.updateWires = function(newWires) {
|
|
|
|
rewiredNodes[node.id] = node;
|
|
|
|
node.newWires = newWires;
|
|
|
|
node.__updateWires(newWires);
|
|
|
|
};
|
|
|
|
}
|
|
|
|
util.inherits(TestErrorNode,Node);
|
|
|
|
|
2015-11-02 15:38:16 +00:00
|
|
|
var TestAsyncNode = function(n) {
|
|
|
|
Node.call(this,n);
|
|
|
|
var node = this;
|
|
|
|
this.scope = n.scope;
|
2019-02-05 14:28:20 +00:00
|
|
|
this.uncaught = n.uncaught;
|
2015-11-02 15:38:16 +00:00
|
|
|
this.foo = n.foo;
|
|
|
|
this.handled = 0;
|
|
|
|
this.messages = [];
|
|
|
|
this.stopped = false;
|
2017-05-15 13:05:33 +01:00
|
|
|
this.closeDelay = n.closeDelay || 50;
|
2015-11-02 15:38:16 +00:00
|
|
|
currentNodes[node.id] = node;
|
|
|
|
this.on('input',function(msg) {
|
|
|
|
node.handled++;
|
2019-07-08 23:23:33 +01:00
|
|
|
msg.handled = node.handled;
|
2015-11-02 15:38:16 +00:00
|
|
|
node.messages.push(msg);
|
|
|
|
node.send(msg);
|
|
|
|
});
|
|
|
|
this.on('close',function(done) {
|
|
|
|
setTimeout(function() {
|
|
|
|
node.stopped = true;
|
|
|
|
stoppedNodes[node.id] = node;
|
2021-02-19 20:44:01 +00:00
|
|
|
stoppedOrder.push(node.id)
|
2015-11-02 15:38:16 +00:00
|
|
|
delete currentNodes[node.id];
|
|
|
|
done();
|
2017-05-15 13:05:33 +01:00
|
|
|
},node.closeDelay);
|
2015-01-15 17:10:32 +00:00
|
|
|
});
|
2015-11-02 15:38:16 +00:00
|
|
|
}
|
|
|
|
util.inherits(TestAsyncNode,Node);
|
2015-10-11 20:37:11 +01:00
|
|
|
|
2019-07-08 23:23:33 +01:00
|
|
|
var TestDoneNode = function(n) {
|
|
|
|
Node.call(this,n);
|
|
|
|
var node = this;
|
|
|
|
this.scope = n.scope;
|
|
|
|
this.uncaught = n.uncaught;
|
|
|
|
this.foo = n.foo;
|
|
|
|
this.handled = 0;
|
|
|
|
this.messages = [];
|
|
|
|
this.stopped = false;
|
|
|
|
this.closeDelay = n.closeDelay || 50;
|
|
|
|
currentNodes[node.id] = node;
|
2019-08-14 22:55:46 +01:00
|
|
|
this.on('input',function(msg, send, done) {
|
2019-07-08 23:23:33 +01:00
|
|
|
node.handled++;
|
|
|
|
node.messages.push(msg);
|
2019-08-14 22:55:46 +01:00
|
|
|
send(msg);
|
2019-07-08 23:23:33 +01:00
|
|
|
done();
|
|
|
|
});
|
|
|
|
this.on('close',function(done) {
|
|
|
|
setTimeout(function() {
|
|
|
|
node.stopped = true;
|
|
|
|
stoppedNodes[node.id] = node;
|
2021-02-19 20:44:01 +00:00
|
|
|
stoppedOrder.push(node.id)
|
2019-07-08 23:23:33 +01:00
|
|
|
delete currentNodes[node.id];
|
|
|
|
done();
|
|
|
|
},node.closeDelay);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
util.inherits(TestDoneNode,Node);
|
|
|
|
|
2015-11-02 15:38:16 +00:00
|
|
|
before(function() {
|
2021-04-09 11:22:57 +01:00
|
|
|
getType = sinon.stub(typeRegistry,"get").callsFake(function(type) {
|
2015-11-02 15:38:16 +00:00
|
|
|
if (type=="test") {
|
|
|
|
return TestNode;
|
2019-01-16 16:27:19 +00:00
|
|
|
} else if (type=="testError"){
|
|
|
|
return TestErrorNode;
|
2019-07-08 23:23:33 +01:00
|
|
|
} else if (type=="testDone"){
|
|
|
|
return TestDoneNode;
|
2015-11-02 15:38:16 +00:00
|
|
|
} else {
|
|
|
|
return TestAsyncNode;
|
2015-01-15 17:10:32 +00:00
|
|
|
}
|
|
|
|
});
|
|
|
|
});
|
2015-11-02 15:38:16 +00:00
|
|
|
after(function() {
|
|
|
|
getType.restore();
|
2015-01-15 17:10:32 +00:00
|
|
|
});
|
2015-10-11 20:37:11 +01:00
|
|
|
|
2015-11-02 15:38:16 +00:00
|
|
|
describe('#constructor',function() {
|
|
|
|
it('called with an empty flow',function() {
|
|
|
|
var config = flowUtils.parseConfig([]);
|
2019-01-16 16:27:19 +00:00
|
|
|
var flow = Flow.create({},config);
|
2015-10-11 20:37:11 +01:00
|
|
|
|
2015-11-02 15:38:16 +00:00
|
|
|
var nodeCount = 0;
|
|
|
|
Object.keys(flow.getActiveNodes()).length.should.equal(0);
|
2015-03-13 17:54:58 +00:00
|
|
|
});
|
2015-11-02 15:38:16 +00:00
|
|
|
});
|
2019-02-05 14:28:20 +00:00
|
|
|
|
2015-11-02 15:38:16 +00:00
|
|
|
describe('#start',function() {
|
2023-06-23 02:11:57 +01:00
|
|
|
it("instantiates an initial configuration and stops it", async function() {
|
2015-11-02 15:38:16 +00:00
|
|
|
var config = flowUtils.parseConfig([
|
|
|
|
{id:"t1",type:"tab"},
|
|
|
|
{id:"1",x:10,y:10,z:"t1",type:"test",foo:"a",wires:["2"]},
|
|
|
|
{id:"2",x:10,y:10,z:"t1",type:"test",foo:"a",wires:["3"]},
|
|
|
|
{id:"3",x:10,y:10,z:"t1",type:"test",foo:"a",wires:[]},
|
|
|
|
{id:"4",z:"t1",type:"test",foo:"a"}
|
|
|
|
]);
|
2019-01-16 16:27:19 +00:00
|
|
|
var flow = Flow.create({},config,config.flows["t1"]);
|
2023-06-23 02:11:57 +01:00
|
|
|
await flow.start();
|
|
|
|
return new Promise((done) => {
|
|
|
|
Object.keys(flow.getActiveNodes()).should.have.length(4);
|
2015-10-11 20:37:11 +01:00
|
|
|
|
2023-06-23 02:11:57 +01:00
|
|
|
flow.getNode('1').should.have.a.property('id','1');
|
|
|
|
flow.getNode('2').should.have.a.property('id','2');
|
|
|
|
flow.getNode('3').should.have.a.property('id','3');
|
|
|
|
flow.getNode('4').should.have.a.property('id','4');
|
2015-11-02 15:38:16 +00:00
|
|
|
|
2023-06-23 02:11:57 +01:00
|
|
|
currentNodes.should.have.a.property("1");
|
|
|
|
currentNodes.should.have.a.property("2");
|
|
|
|
currentNodes.should.have.a.property("3");
|
|
|
|
currentNodes.should.have.a.property("4");
|
2015-10-11 20:37:11 +01:00
|
|
|
|
2023-06-23 02:11:57 +01:00
|
|
|
currentNodes["1"].should.have.a.property("handled",0);
|
|
|
|
currentNodes["2"].should.have.a.property("handled",0);
|
|
|
|
currentNodes["3"].should.have.a.property("handled",0);
|
2015-10-11 20:37:11 +01:00
|
|
|
|
2023-06-23 02:11:57 +01:00
|
|
|
currentNodes["3"].on("input", function() {
|
|
|
|
currentNodes["1"].should.have.a.property("handled",1);
|
|
|
|
currentNodes["2"].should.have.a.property("handled",1);
|
|
|
|
currentNodes["3"].should.have.a.property("handled",1);
|
2019-07-08 23:23:33 +01:00
|
|
|
|
2023-06-23 02:11:57 +01:00
|
|
|
flow.stop().then(function() {
|
|
|
|
try {
|
|
|
|
currentNodes.should.not.have.a.property("1");
|
|
|
|
currentNodes.should.not.have.a.property("2");
|
|
|
|
currentNodes.should.not.have.a.property("3");
|
|
|
|
currentNodes.should.not.have.a.property("4");
|
|
|
|
stoppedNodes.should.have.a.property("1");
|
|
|
|
stoppedNodes.should.have.a.property("2");
|
|
|
|
stoppedNodes.should.have.a.property("3");
|
|
|
|
stoppedNodes.should.have.a.property("4");
|
|
|
|
done();
|
|
|
|
} catch(err) {
|
|
|
|
done(err);
|
|
|
|
}
|
|
|
|
});
|
2019-07-08 23:23:33 +01:00
|
|
|
});
|
2023-06-23 02:11:57 +01:00
|
|
|
currentNodes["1"].receive({payload:"test"});
|
|
|
|
})
|
2015-03-13 17:54:58 +00:00
|
|
|
});
|
|
|
|
|
2023-06-23 02:11:57 +01:00
|
|
|
it("instantiates config nodes in the right order",async function() {
|
2016-04-27 10:31:19 +01:00
|
|
|
var config = flowUtils.parseConfig([
|
|
|
|
{id:"t1",type:"tab"},
|
|
|
|
{id:"1",x:10,y:10,z:"t1",type:"test",foo:"a",wires:["2"]},
|
|
|
|
{id:"2",x:10,y:10,z:"t1",type:"test",foo:"a",wires:["3"]},
|
|
|
|
{id:"3",x:10,y:10,z:"t1",type:"test",foo:"a",wires:[]},
|
|
|
|
{id:"4",z:"t1",type:"test",foo:"5"}, // This node depends on #5
|
|
|
|
{id:"5",z:"t1",type:"test"}
|
|
|
|
]);
|
2019-01-16 16:27:19 +00:00
|
|
|
var flow = Flow.create({},config,config.flows["t1"]);
|
2023-06-23 02:11:57 +01:00
|
|
|
await flow.start();
|
2016-04-27 10:31:19 +01:00
|
|
|
Object.keys(flow.getActiveNodes()).should.have.length(5);
|
|
|
|
currentNodes.should.have.a.property("1");
|
|
|
|
currentNodes.should.have.a.property("2");
|
|
|
|
currentNodes.should.have.a.property("3");
|
|
|
|
currentNodes.should.have.a.property("4");
|
|
|
|
currentNodes.should.have.a.property("5");
|
|
|
|
|
|
|
|
currentNodes["1"].should.have.a.property("_index",2);
|
|
|
|
currentNodes["2"].should.have.a.property("_index",3);
|
|
|
|
currentNodes["3"].should.have.a.property("_index",4);
|
|
|
|
currentNodes["4"].should.have.a.property("_index",1);
|
|
|
|
currentNodes["5"].should.have.a.property("_index",0);
|
|
|
|
|
2023-06-23 02:11:57 +01:00
|
|
|
return flow.stop().then(function() {
|
2016-04-27 10:31:19 +01:00
|
|
|
currentNodes.should.not.have.a.property("1");
|
|
|
|
currentNodes.should.not.have.a.property("2");
|
|
|
|
currentNodes.should.not.have.a.property("3");
|
|
|
|
currentNodes.should.not.have.a.property("4");
|
|
|
|
currentNodes.should.not.have.a.property("5");
|
|
|
|
stoppedNodes.should.have.a.property("1");
|
|
|
|
stoppedNodes.should.have.a.property("2");
|
|
|
|
stoppedNodes.should.have.a.property("3");
|
|
|
|
stoppedNodes.should.have.a.property("4");
|
|
|
|
stoppedNodes.should.have.a.property("5");
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
|
2023-06-23 02:11:57 +01:00
|
|
|
it("detects dependency loops in config nodes",async function() {
|
2016-04-27 10:31:19 +01:00
|
|
|
var config = flowUtils.parseConfig([
|
|
|
|
{id:"t1",type:"tab"},
|
|
|
|
{id:"node1",z:"t1",type:"test",foo:"node2"}, // This node depends on #5
|
|
|
|
{id:"node2",z:"t1",type:"test",foo:"node1"}
|
|
|
|
]);
|
2019-01-16 16:27:19 +00:00
|
|
|
var flow = Flow.create({},config,config.flows["t1"]);
|
2016-04-27 10:31:19 +01:00
|
|
|
/*jshint immed: false */
|
2023-06-23 02:11:57 +01:00
|
|
|
return flow.start().catch(err => {
|
|
|
|
err.toString().should.equal("Error: Circular config node dependency detected: node1")
|
|
|
|
})
|
2016-04-27 10:31:19 +01:00
|
|
|
});
|
2015-11-02 15:38:16 +00:00
|
|
|
|
2023-06-23 02:11:57 +01:00
|
|
|
it("rewires nodes specified by diff", async function() {
|
2015-11-02 15:38:16 +00:00
|
|
|
var config = flowUtils.parseConfig([
|
|
|
|
{id:"t1",type:"tab"},
|
|
|
|
{id:"1",x:10,y:10,z:"t1",type:"test",foo:"a",wires:["2"]},
|
|
|
|
{id:"2",x:10,y:10,z:"t1",type:"test",foo:"a",wires:["3"]},
|
|
|
|
{id:"3",x:10,y:10,z:"t1",type:"test",foo:"a",wires:[]}
|
|
|
|
]);
|
|
|
|
|
2019-01-16 16:27:19 +00:00
|
|
|
var flow = Flow.create({},config,config.flows["t1"]);
|
2015-03-13 17:54:58 +00:00
|
|
|
createCount.should.equal(0);
|
2023-06-23 02:11:57 +01:00
|
|
|
await flow.start();
|
2015-11-02 15:38:16 +00:00
|
|
|
//TODO: use update to pass in new wiring and verify the change
|
2015-03-13 17:54:58 +00:00
|
|
|
createCount.should.equal(3);
|
2015-11-02 15:38:16 +00:00
|
|
|
flow.start({rewired:["2"]});
|
2015-03-13 17:54:58 +00:00
|
|
|
createCount.should.equal(3);
|
|
|
|
rewiredNodes.should.have.a.property("2");
|
|
|
|
});
|
|
|
|
|
2023-06-23 02:11:57 +01:00
|
|
|
it("instantiates a node with environment variable property values", async function() {
|
2015-11-02 15:38:16 +00:00
|
|
|
after(function() {
|
|
|
|
delete process.env.NODE_RED_TEST_VALUE;
|
|
|
|
})
|
|
|
|
process.env.NODE_RED_TEST_VALUE = "a-value";
|
|
|
|
var config = flowUtils.parseConfig([
|
|
|
|
{id:"t1",type:"tab"},
|
|
|
|
{id:"1",x:10,y:10,z:"t1",type:"test",foo:"$(NODE_RED_TEST_VALUE)",wires:[]},
|
|
|
|
{id:"2",x:10,y:10,z:"t1",type:"test",foo:{a:"$(NODE_RED_TEST_VALUE)"},wires:[]},
|
|
|
|
{id:"3",x:10,y:10,z:"t1",type:"test",foo:" $(NODE_RED_TEST_VALUE)",wires:[]},
|
|
|
|
{id:"4",x:10,y:10,z:"t1",type:"test",foo:"$(NODE_RED_TEST_VALUE) ",wires:[]},
|
|
|
|
{id:"5",x:10,y:10,z:"t1",type:"test",foo:"$(NODE_RED_TEST_VALUE_NONE)",wires:[]},
|
|
|
|
{id:"6",x:10,y:10,z:"t1",type:"test",foo:["$(NODE_RED_TEST_VALUE)"],wires:[]}
|
|
|
|
]);
|
2019-01-16 22:38:04 +00:00
|
|
|
var flow = Flow.create({getSetting:v=>process.env[v]},config,config.flows["t1"]);
|
2023-06-23 02:11:57 +01:00
|
|
|
await flow.start();
|
2015-11-02 15:38:16 +00:00
|
|
|
|
|
|
|
var activeNodes = flow.getActiveNodes();
|
|
|
|
|
|
|
|
activeNodes["1"].foo.should.equal("a-value");
|
|
|
|
activeNodes["2"].foo.a.should.equal("a-value");
|
|
|
|
activeNodes["3"].foo.should.equal(" $(NODE_RED_TEST_VALUE)");
|
|
|
|
activeNodes["4"].foo.should.equal("$(NODE_RED_TEST_VALUE) ");
|
|
|
|
activeNodes["5"].foo.should.equal("$(NODE_RED_TEST_VALUE_NONE)");
|
|
|
|
activeNodes["6"].foo[0].should.equal("a-value");
|
|
|
|
|
2023-06-23 02:11:57 +01:00
|
|
|
await flow.stop()
|
2015-03-13 17:54:58 +00:00
|
|
|
});
|
2015-10-11 20:37:11 +01:00
|
|
|
|
2023-06-23 02:11:57 +01:00
|
|
|
it("ignores disabled nodes", async function() {
|
2019-06-18 11:02:31 +01:00
|
|
|
var config = flowUtils.parseConfig([
|
|
|
|
{id:"t1",type:"tab"},
|
|
|
|
{id:"1",x:10,y:10,z:"t1",type:"test",foo:"a",wires:["2"]},
|
|
|
|
{id:"2",x:10,y:10,z:"t1",d:true,type:"test",foo:"a",wires:["3"]},
|
|
|
|
{id:"3",x:10,y:10,z:"t1",type:"test",foo:"a",wires:[]},
|
|
|
|
{id:"4",z:"t1",type:"test",foo:"a"},
|
|
|
|
{id:"5",z:"t1",type:"test",d:true,foo:"a"}
|
|
|
|
|
|
|
|
]);
|
|
|
|
var flow = Flow.create({},config,config.flows["t1"]);
|
2023-06-23 02:11:57 +01:00
|
|
|
await flow.start();
|
2019-06-18 11:02:31 +01:00
|
|
|
|
|
|
|
Object.keys(flow.getActiveNodes()).should.have.length(3);
|
|
|
|
|
|
|
|
flow.getNode('1').should.have.a.property('id','1');
|
|
|
|
should.not.exist(flow.getNode('2'));
|
|
|
|
flow.getNode('3').should.have.a.property('id','3');
|
|
|
|
flow.getNode('4').should.have.a.property('id','4');
|
|
|
|
should.not.exist(flow.getNode('5'));
|
|
|
|
|
|
|
|
currentNodes.should.have.a.property("1");
|
|
|
|
currentNodes.should.not.have.a.property("2");
|
|
|
|
currentNodes.should.have.a.property("3");
|
|
|
|
currentNodes.should.have.a.property("4");
|
|
|
|
|
|
|
|
currentNodes["1"].should.have.a.property("handled",0);
|
|
|
|
currentNodes["3"].should.have.a.property("handled",0);
|
|
|
|
|
|
|
|
currentNodes["1"].receive({payload:"test"});
|
2023-06-23 02:11:57 +01:00
|
|
|
await NR_TEST_UTILS.sleep(50)
|
|
|
|
currentNodes["1"].should.have.a.property("handled",1);
|
|
|
|
// Message doesn't reach 3 as 2 is disabled
|
|
|
|
currentNodes["3"].should.have.a.property("handled",0);
|
2019-06-18 11:02:31 +01:00
|
|
|
|
2023-06-23 02:11:57 +01:00
|
|
|
await flow.stop()
|
|
|
|
currentNodes.should.not.have.a.property("1");
|
|
|
|
currentNodes.should.not.have.a.property("2");
|
|
|
|
currentNodes.should.not.have.a.property("3");
|
|
|
|
currentNodes.should.not.have.a.property("4");
|
|
|
|
stoppedNodes.should.have.a.property("1");
|
|
|
|
stoppedNodes.should.not.have.a.property("2");
|
|
|
|
stoppedNodes.should.have.a.property("3");
|
|
|
|
stoppedNodes.should.have.a.property("4");
|
2019-06-18 11:02:31 +01:00
|
|
|
});
|
2015-11-02 15:38:16 +00:00
|
|
|
|
|
|
|
});
|
|
|
|
|
|
|
|
describe('#stop', function() {
|
|
|
|
|
|
|
|
|
2023-06-23 02:11:57 +01:00
|
|
|
it("stops all nodes", async function() {
|
2015-11-02 15:38:16 +00:00
|
|
|
var config = flowUtils.parseConfig([
|
|
|
|
{id:"t1",type:"tab"},
|
|
|
|
{id:"1",x:10,y:10,z:"t1",type:"test",foo:"a",wires:["2"]},
|
|
|
|
{id:"2",x:10,y:10,z:"t1",type:"test",foo:"a",wires:["3"]},
|
|
|
|
{id:"3",x:10,y:10,z:"t1",type:"asyncTest",foo:"a",wires:[]}
|
|
|
|
]);
|
2019-01-16 16:27:19 +00:00
|
|
|
var flow = Flow.create({},config,config.flows["t1"]);
|
2023-06-23 02:11:57 +01:00
|
|
|
await flow.start();
|
2015-03-13 17:54:58 +00:00
|
|
|
currentNodes.should.have.a.property("1");
|
|
|
|
currentNodes.should.have.a.property("2");
|
|
|
|
currentNodes.should.have.a.property("3");
|
2015-10-11 20:37:11 +01:00
|
|
|
|
2023-06-23 02:11:57 +01:00
|
|
|
await flow.stop()
|
|
|
|
currentNodes.should.not.have.a.property("1");
|
|
|
|
currentNodes.should.not.have.a.property("2");
|
|
|
|
currentNodes.should.not.have.a.property("3");
|
|
|
|
stoppedNodes.should.have.a.property("1");
|
|
|
|
stoppedNodes.should.have.a.property("2");
|
|
|
|
stoppedNodes.should.have.a.property("3");
|
2015-03-13 17:54:58 +00:00
|
|
|
});
|
2015-10-11 20:37:11 +01:00
|
|
|
|
2023-06-23 02:11:57 +01:00
|
|
|
it("stops specified nodes", async function() {
|
2015-11-02 15:38:16 +00:00
|
|
|
var config = flowUtils.parseConfig([
|
|
|
|
{id:"t1",type:"tab"},
|
|
|
|
{id:"1",x:10,y:10,z:"t1",type:"test",foo:"a",wires:["2"]},
|
|
|
|
{id:"2",x:10,y:10,z:"t1",type:"test",foo:"a",wires:["3"]},
|
|
|
|
{id:"3",x:10,y:10,z:"t1",type:"test",foo:"a",wires:[]}
|
|
|
|
]);
|
2019-01-16 16:27:19 +00:00
|
|
|
var flow = Flow.create({},config,config.flows["t1"]);
|
2023-06-23 02:11:57 +01:00
|
|
|
await flow.start();
|
2015-03-13 17:54:58 +00:00
|
|
|
currentNodes.should.have.a.property("1");
|
|
|
|
currentNodes.should.have.a.property("2");
|
|
|
|
currentNodes.should.have.a.property("3");
|
2015-10-11 20:37:11 +01:00
|
|
|
|
2023-06-23 02:11:57 +01:00
|
|
|
await flow.stop(["2"])
|
|
|
|
currentNodes.should.have.a.property("1");
|
|
|
|
currentNodes.should.not.have.a.property("2");
|
|
|
|
currentNodes.should.have.a.property("3");
|
|
|
|
stoppedNodes.should.not.have.a.property("1");
|
|
|
|
stoppedNodes.should.have.a.property("2");
|
|
|
|
stoppedNodes.should.not.have.a.property("3");
|
2015-03-13 17:54:58 +00:00
|
|
|
});
|
|
|
|
|
2023-06-23 02:11:57 +01:00
|
|
|
it("stops config nodes last", async function() {
|
2021-02-19 20:44:01 +00:00
|
|
|
var config = flowUtils.parseConfig([
|
|
|
|
{id:"t1",type:"tab"},
|
|
|
|
{id:"1",x:10,y:10,z:"t1",type:"test",foo:"a",wires:["2"]},
|
|
|
|
{id:"c1",z:"t1",type:"test"},
|
|
|
|
{id:"2",x:10,y:10,z:"t1",type:"test",foo:"a",wires:["3"]},
|
|
|
|
{id:"c2",z:"t1",type:"test"},
|
|
|
|
{id:"3",x:10,y:10,z:"t1",type:"test",foo:"a",wires:[]},
|
|
|
|
{id:"c3",z:"t1",type:"test"}
|
|
|
|
]);
|
|
|
|
var flow = Flow.create({},config,config.flows["t1"]);
|
2023-06-23 02:11:57 +01:00
|
|
|
await flow.start();
|
2021-02-19 20:44:01 +00:00
|
|
|
|
|
|
|
currentNodes.should.have.a.property("1");
|
|
|
|
currentNodes.should.have.a.property("2");
|
|
|
|
currentNodes.should.have.a.property("3");
|
|
|
|
currentNodes.should.have.a.property("c1");
|
|
|
|
currentNodes.should.have.a.property("c2");
|
|
|
|
currentNodes.should.have.a.property("c3");
|
|
|
|
stoppedOrder.should.have.a.length(0);
|
|
|
|
|
2023-06-23 02:11:57 +01:00
|
|
|
await flow.stop()
|
|
|
|
stoppedOrder.should.eql([ '1', '2', '3', 'c1', 'c2', 'c3' ]);
|
2021-02-19 20:44:01 +00:00
|
|
|
});
|
|
|
|
|
|
|
|
|
2023-06-23 02:11:57 +01:00
|
|
|
it("Times out a node that fails to close", async function() {
|
2018-04-23 14:24:51 +01:00
|
|
|
Flow.init({settings:{nodeCloseTimeout:50},log:{
|
|
|
|
log: sinon.stub(),
|
|
|
|
debug: sinon.stub(),
|
|
|
|
trace: sinon.stub(),
|
|
|
|
warn: sinon.stub(),
|
|
|
|
info: sinon.stub(),
|
|
|
|
metric: sinon.stub(),
|
|
|
|
_: function() { return "abc"}
|
|
|
|
}});
|
2017-05-15 13:05:33 +01:00
|
|
|
var config = flowUtils.parseConfig([
|
|
|
|
{id:"t1",type:"tab"},
|
|
|
|
{id:"1",x:10,y:10,z:"t1",type:"testAsync",closeDelay: 80, foo:"a",wires:["2"]},
|
|
|
|
{id:"2",x:10,y:10,z:"t1",type:"test",foo:"a",wires:["3"]},
|
|
|
|
{id:"3",x:10,y:10,z:"t1",type:"test",foo:"a",wires:[]}
|
|
|
|
]);
|
2019-01-16 16:27:19 +00:00
|
|
|
var flow = Flow.create({},config,config.flows["t1"]);
|
2023-06-23 02:11:57 +01:00
|
|
|
await flow.start();
|
2017-05-15 13:05:33 +01:00
|
|
|
|
|
|
|
currentNodes.should.have.a.property("1");
|
|
|
|
currentNodes.should.have.a.property("2");
|
|
|
|
currentNodes.should.have.a.property("3");
|
|
|
|
|
2023-06-23 02:11:57 +01:00
|
|
|
await flow.stop()
|
|
|
|
currentNodes.should.have.a.property("1");
|
|
|
|
currentNodes.should.not.have.a.property("2");
|
|
|
|
currentNodes.should.not.have.a.property("3");
|
|
|
|
stoppedNodes.should.not.have.a.property("1");
|
|
|
|
stoppedNodes.should.have.a.property("2");
|
|
|
|
stoppedNodes.should.have.a.property("3");
|
|
|
|
await NR_TEST_UTILS.sleep(40)
|
|
|
|
currentNodes.should.not.have.a.property("1");
|
|
|
|
stoppedNodes.should.have.a.property("1");
|
2017-05-15 13:05:33 +01:00
|
|
|
});
|
2015-11-02 15:38:16 +00:00
|
|
|
|
|
|
|
});
|
2019-02-05 14:28:20 +00:00
|
|
|
|
2019-01-29 21:49:20 +00:00
|
|
|
describe('#getNode',function() {
|
2023-06-23 02:11:57 +01:00
|
|
|
it("gets a node known to the flow", async function() {
|
2019-01-29 21:49:20 +00:00
|
|
|
var config = flowUtils.parseConfig([
|
|
|
|
{id:"t1",type:"tab"},
|
|
|
|
{id:"1",x:10,y:10,z:"t1",type:"test",foo:"a",wires:["2"]},
|
|
|
|
{id:"2",x:10,y:10,z:"t1",type:"test",foo:"a",wires:["3"]},
|
|
|
|
{id:"3",x:10,y:10,z:"t1",type:"test",foo:"a",wires:[]},
|
|
|
|
{id:"4",z:"t1",type:"test",foo:"a"}
|
|
|
|
]);
|
|
|
|
var flow = Flow.create({},config,config.flows["t1"]);
|
2023-06-23 02:11:57 +01:00
|
|
|
await flow.start();
|
2019-01-29 21:49:20 +00:00
|
|
|
Object.keys(flow.getActiveNodes()).should.have.length(4);
|
|
|
|
flow.getNode('1').should.have.a.property('id','1');
|
2023-06-23 02:11:57 +01:00
|
|
|
await flow.stop();
|
2019-01-29 21:49:20 +00:00
|
|
|
});
|
|
|
|
|
2023-06-23 02:11:57 +01:00
|
|
|
it("passes to parent if node not known locally", async function() {
|
2019-01-29 21:49:20 +00:00
|
|
|
var config = flowUtils.parseConfig([
|
|
|
|
{id:"t1",type:"tab"},
|
|
|
|
{id:"1",x:10,y:10,z:"t1",type:"test",foo:"a",wires:["2"]},
|
|
|
|
{id:"2",x:10,y:10,z:"t1",type:"test",foo:"a",wires:["3"]},
|
|
|
|
{id:"3",x:10,y:10,z:"t1",type:"test",foo:"a",wires:[]},
|
|
|
|
{id:"4",z:"t1",type:"test",foo:"a"}
|
|
|
|
]);
|
|
|
|
var flow = Flow.create({
|
|
|
|
getNode: id => { return {id:id}}
|
|
|
|
},config,config.flows["t1"]);
|
2023-06-23 02:11:57 +01:00
|
|
|
await flow.start();
|
2019-01-29 21:49:20 +00:00
|
|
|
Object.keys(flow.getActiveNodes()).should.have.length(4);
|
|
|
|
flow.getNode('1').should.have.a.property('id','1');
|
|
|
|
flow.getNode('parentNode').should.have.a.property('id','parentNode');
|
2023-06-23 02:11:57 +01:00
|
|
|
await flow.stop()
|
2019-01-29 21:49:20 +00:00
|
|
|
});
|
|
|
|
|
2023-06-23 02:11:57 +01:00
|
|
|
it("does not pass to parent if cancelBubble set", async function() {
|
2019-01-29 21:49:20 +00:00
|
|
|
var config = flowUtils.parseConfig([
|
|
|
|
{id:"t1",type:"tab"},
|
|
|
|
{id:"1",x:10,y:10,z:"t1",type:"test",foo:"a",wires:["2"]},
|
|
|
|
{id:"2",x:10,y:10,z:"t1",type:"test",foo:"a",wires:["3"]},
|
|
|
|
{id:"3",x:10,y:10,z:"t1",type:"test",foo:"a",wires:[]},
|
|
|
|
{id:"4",z:"t1",type:"test",foo:"a"}
|
|
|
|
]);
|
|
|
|
var flow = Flow.create({
|
|
|
|
getNode: id => { return {id:id}}
|
|
|
|
},config,config.flows["t1"]);
|
2023-06-23 02:11:57 +01:00
|
|
|
await flow.start();
|
2019-01-29 21:49:20 +00:00
|
|
|
Object.keys(flow.getActiveNodes()).should.have.length(4);
|
|
|
|
flow.getNode('1').should.have.a.property('id','1');
|
|
|
|
should.not.exist(flow.getNode('parentNode',true));
|
2023-06-23 02:11:57 +01:00
|
|
|
await flow.stop()
|
2019-01-29 21:49:20 +00:00
|
|
|
});
|
|
|
|
});
|
2015-11-02 15:38:16 +00:00
|
|
|
|
|
|
|
describe("#handleStatus",function() {
|
2023-06-23 02:11:57 +01:00
|
|
|
it("passes a status event to the adjacent status node", async function() {
|
2015-11-02 15:38:16 +00:00
|
|
|
var config = flowUtils.parseConfig([
|
|
|
|
{id:"t1",type:"tab"},
|
|
|
|
{id:"1",x:10,y:10,z:"t1",type:"test",name:"a",wires:["2"]},
|
|
|
|
{id:"2",x:10,y:10,z:"t1",type:"test",wires:["3"]},
|
|
|
|
{id:"3",x:10,y:10,z:"t1",type:"test",foo:"a",wires:[]},
|
|
|
|
{id:"sn",x:10,y:10,z:"t1",type:"status",foo:"a",wires:[]},
|
|
|
|
{id:"sn2",x:10,y:10,z:"t1",type:"status",foo:"a",wires:[]}
|
|
|
|
]);
|
2019-01-16 16:27:19 +00:00
|
|
|
var flow = Flow.create({},config,config.flows["t1"]);
|
2015-10-11 20:37:11 +01:00
|
|
|
|
2023-06-23 02:11:57 +01:00
|
|
|
await flow.start();
|
2015-10-11 20:37:11 +01:00
|
|
|
|
2015-11-02 15:38:16 +00:00
|
|
|
var activeNodes = flow.getActiveNodes();
|
|
|
|
Object.keys(activeNodes).should.have.length(5);
|
2015-03-13 17:54:58 +00:00
|
|
|
|
|
|
|
|
2019-03-04 10:23:10 +00:00
|
|
|
flow.handleStatus(config.flows["t1"].nodes["1"],{text:"my-status",random:"otherProperty"});
|
2023-06-23 02:11:57 +01:00
|
|
|
await NR_TEST_UTILS.sleep(50)
|
|
|
|
currentNodes["sn"].should.have.a.property("handled",1);
|
|
|
|
var statusMessage = currentNodes["sn"].messages[0];
|
|
|
|
statusMessage.should.have.a.property("status");
|
|
|
|
statusMessage.status.should.have.a.property("text","my-status");
|
|
|
|
statusMessage.status.should.have.a.property("source");
|
|
|
|
statusMessage.status.source.should.have.a.property("id","1");
|
|
|
|
statusMessage.status.source.should.have.a.property("type","test");
|
|
|
|
statusMessage.status.source.should.have.a.property("name","a");
|
|
|
|
currentNodes["sn2"].should.have.a.property("handled",1);
|
|
|
|
statusMessage = currentNodes["sn2"].messages[0];
|
|
|
|
statusMessage.should.have.a.property("status");
|
|
|
|
statusMessage.status.should.have.a.property("text","my-status");
|
|
|
|
statusMessage.status.should.have.a.property("random","otherProperty");
|
|
|
|
statusMessage.status.should.have.a.property("source");
|
|
|
|
statusMessage.status.source.should.have.a.property("id","1");
|
|
|
|
statusMessage.status.source.should.have.a.property("type","test");
|
|
|
|
statusMessage.status.source.should.have.a.property("name","a");
|
|
|
|
await flow.stop()
|
2015-11-02 15:38:16 +00:00
|
|
|
});
|
2023-06-23 02:11:57 +01:00
|
|
|
it("passes a status event to the adjacent scoped status node ", async function() {
|
2015-11-02 15:38:16 +00:00
|
|
|
var config = flowUtils.parseConfig([
|
|
|
|
{id:"t1",type:"tab"},
|
|
|
|
{id:"1",x:10,y:10,z:"t1",type:"test",name:"a",wires:["2"]},
|
|
|
|
{id:"2",x:10,y:10,z:"t1",type:"test",wires:["3"]},
|
|
|
|
{id:"3",x:10,y:10,z:"t1",type:"test",foo:"a",wires:[]},
|
|
|
|
{id:"sn",x:10,y:10,z:"t1",type:"status",scope:["2"],foo:"a",wires:[]},
|
|
|
|
{id:"sn2",x:10,y:10,z:"t1",type:"status",scope:["1"],foo:"a",wires:[]}
|
|
|
|
]);
|
2019-01-16 16:27:19 +00:00
|
|
|
var flow = Flow.create({},config,config.flows["t1"]);
|
2015-10-11 20:37:11 +01:00
|
|
|
|
2023-06-23 02:11:57 +01:00
|
|
|
await flow.start();
|
2015-11-02 15:38:16 +00:00
|
|
|
|
|
|
|
var activeNodes = flow.getActiveNodes();
|
|
|
|
Object.keys(activeNodes).should.have.length(5);
|
|
|
|
|
|
|
|
flow.handleStatus(config.flows["t1"].nodes["1"],{text:"my-status"});
|
|
|
|
|
2023-06-23 02:11:57 +01:00
|
|
|
await NR_TEST_UTILS.sleep(50)
|
|
|
|
currentNodes["sn"].should.have.a.property("handled",0);
|
|
|
|
currentNodes["sn2"].should.have.a.property("handled",1);
|
|
|
|
var statusMessage = currentNodes["sn2"].messages[0];
|
|
|
|
statusMessage.should.have.a.property("status");
|
|
|
|
statusMessage.status.should.have.a.property("text","my-status");
|
|
|
|
statusMessage.status.should.have.a.property("source");
|
|
|
|
statusMessage.status.source.should.have.a.property("id","1");
|
|
|
|
statusMessage.status.source.should.have.a.property("type","test");
|
|
|
|
statusMessage.status.source.should.have.a.property("name","a");
|
|
|
|
await flow.stop()
|
2015-01-16 15:43:47 +00:00
|
|
|
});
|
2015-11-02 15:38:16 +00:00
|
|
|
|
2023-06-23 02:11:57 +01:00
|
|
|
it("passes a status event to the group scoped status node", async function() {
|
2023-05-22 22:33:31 +01:00
|
|
|
var config = flowUtils.parseConfig([
|
|
|
|
{id:"t1",type:"tab"},
|
2023-06-23 02:11:57 +01:00
|
|
|
{id: "g1", type: "group", g: "g3", z:"t1" },
|
|
|
|
{id: "g2", type: "group", z:"t1" },
|
|
|
|
{id: "g3", type: "group", z:"t1" },
|
2023-05-22 22:33:31 +01:00
|
|
|
{id:"1",x:10,y:10,z:"t1",g:"g1", type:"test",name:"a",wires:["2"]},
|
|
|
|
// sn - in the same group as source node
|
|
|
|
{id:"sn",x:10,y:10,z:"t1",g:"g1", type:"status",scope:"group",wires:[]},
|
|
|
|
// sn2 - in a different group hierarchy to the source node
|
|
|
|
{id:"sn2",x:10,y:10,z:"t1", g:"g2", type:"status",scope:"group",wires:[]},
|
|
|
|
// sn3 - in a higher-level group to the source node
|
|
|
|
{id:"sn3",x:10,y:10,z:"t1", g:"g3", type:"status",scope:"group",wires:[]},
|
|
|
|
// sn2 - in a different group hierarchy, but not scope to the group
|
|
|
|
{id:"sn4",x:10,y:10,z:"t1", g:"g2", type:"status",wires:[]},
|
|
|
|
|
|
|
|
]);
|
|
|
|
var flow = Flow.create({},config,config.flows["t1"]);
|
|
|
|
|
2023-06-23 02:11:57 +01:00
|
|
|
await flow.start();
|
2023-05-22 22:33:31 +01:00
|
|
|
|
|
|
|
var activeNodes = flow.getActiveNodes();
|
|
|
|
flow.handleStatus(config.flows["t1"].nodes["1"],{text:"my-status"});
|
2023-06-23 02:11:57 +01:00
|
|
|
await NR_TEST_UTILS.sleep(50)
|
|
|
|
currentNodes["sn"].should.have.a.property("handled",1);
|
|
|
|
currentNodes["sn2"].should.have.a.property("handled",0);
|
|
|
|
currentNodes["sn3"].should.have.a.property("handled",1);
|
|
|
|
currentNodes["sn3"].should.have.a.property("handled",1);
|
|
|
|
await flow.stop()
|
2023-05-22 22:33:31 +01:00
|
|
|
});
|
2015-11-02 15:38:16 +00:00
|
|
|
});
|
2015-01-16 15:43:47 +00:00
|
|
|
|
2015-11-02 15:38:16 +00:00
|
|
|
describe("#handleError",function() {
|
2023-06-23 02:11:57 +01:00
|
|
|
it("passes an error event to the adjacent catch node", async function() {
|
2015-11-02 15:38:16 +00:00
|
|
|
var config = flowUtils.parseConfig([
|
|
|
|
{id:"t1",type:"tab"},
|
|
|
|
{id:"1",x:10,y:10,z:"t1",type:"test",name:"a",wires:["2"]},
|
|
|
|
{id:"2",x:10,y:10,z:"t1",type:"test",wires:["3"]},
|
|
|
|
{id:"3",x:10,y:10,z:"t1",type:"test",foo:"a",wires:[]},
|
|
|
|
{id:"sn",x:10,y:10,z:"t1",type:"catch",foo:"a",wires:[]},
|
2019-02-05 14:28:20 +00:00
|
|
|
{id:"sn2",x:10,y:10,z:"t1",type:"catch",foo:"a",wires:[]},
|
|
|
|
{id:"sn3",x:10,y:10,z:"t1",type:"catch",uncaught:true,wires:[]}
|
2015-11-02 15:38:16 +00:00
|
|
|
]);
|
2019-01-16 16:27:19 +00:00
|
|
|
var flow = Flow.create({},config,config.flows["t1"]);
|
2015-11-02 15:38:16 +00:00
|
|
|
|
2023-06-23 02:11:57 +01:00
|
|
|
await flow.start();
|
2015-10-11 20:37:11 +01:00
|
|
|
|
2015-11-02 15:38:16 +00:00
|
|
|
var activeNodes = flow.getActiveNodes();
|
2019-02-05 14:28:20 +00:00
|
|
|
Object.keys(activeNodes).should.have.length(6);
|
2015-11-02 15:38:16 +00:00
|
|
|
flow.handleError(config.flows["t1"].nodes["1"],"my-error",{a:"foo"});
|
2023-06-23 02:11:57 +01:00
|
|
|
await NR_TEST_UTILS.sleep(50)
|
|
|
|
currentNodes["sn"].should.have.a.property("handled",1);
|
|
|
|
var statusMessage = currentNodes["sn"].messages[0];
|
|
|
|
|
|
|
|
statusMessage.should.have.a.property("error");
|
|
|
|
statusMessage.error.should.have.a.property("message","my-error");
|
|
|
|
statusMessage.error.should.have.a.property("source");
|
|
|
|
statusMessage.error.source.should.have.a.property("id","1");
|
|
|
|
statusMessage.error.source.should.have.a.property("type","test");
|
|
|
|
statusMessage.error.source.should.have.a.property("name","a");
|
|
|
|
|
|
|
|
currentNodes["sn2"].should.have.a.property("handled",1);
|
|
|
|
statusMessage = currentNodes["sn2"].messages[0];
|
|
|
|
|
|
|
|
statusMessage.should.have.a.property("error");
|
|
|
|
statusMessage.error.should.have.a.property("message","my-error");
|
|
|
|
statusMessage.error.should.have.a.property("source");
|
|
|
|
statusMessage.error.source.should.have.a.property("id","1");
|
|
|
|
statusMessage.error.source.should.have.a.property("type","test");
|
|
|
|
statusMessage.error.source.should.have.a.property("name","a");
|
|
|
|
|
|
|
|
// Node sn3 has uncaught:true - so should not get called
|
|
|
|
currentNodes["sn3"].should.have.a.property("handled",0);
|
|
|
|
await flow.stop()
|
2015-01-16 15:43:47 +00:00
|
|
|
});
|
2023-06-23 02:11:57 +01:00
|
|
|
|
|
|
|
it("passes an error event to the adjacent scoped catch node ", async function() {
|
2015-11-02 15:38:16 +00:00
|
|
|
var config = flowUtils.parseConfig([
|
|
|
|
{id:"t1",type:"tab"},
|
|
|
|
{id:"1",x:10,y:10,z:"t1",type:"test",name:"a",wires:["2"]},
|
|
|
|
{id:"2",x:10,y:10,z:"t1",type:"test",wires:["3"]},
|
|
|
|
{id:"3",x:10,y:10,z:"t1",type:"test",foo:"a",wires:[]},
|
|
|
|
{id:"sn",x:10,y:10,z:"t1",type:"catch",scope:["2"],foo:"a",wires:[]},
|
2019-02-05 14:28:20 +00:00
|
|
|
{id:"sn2",x:10,y:10,z:"t1",type:"catch",scope:["1"],foo:"a",wires:[]},
|
|
|
|
{id:"sn3",x:10,y:10,z:"t1",type:"catch",uncaught:true,wires:[]},
|
|
|
|
{id:"sn4",x:10,y:10,z:"t1",type:"catch",uncaught:true,wires:[]}
|
2015-11-02 15:38:16 +00:00
|
|
|
]);
|
2019-01-16 16:27:19 +00:00
|
|
|
var flow = Flow.create({},config,config.flows["t1"]);
|
2015-10-11 20:37:11 +01:00
|
|
|
|
2023-06-23 02:11:57 +01:00
|
|
|
await flow.start();
|
2015-10-11 20:37:11 +01:00
|
|
|
|
2015-11-02 15:38:16 +00:00
|
|
|
var activeNodes = flow.getActiveNodes();
|
2019-02-05 14:28:20 +00:00
|
|
|
Object.keys(activeNodes).should.have.length(7);
|
2015-10-11 20:37:11 +01:00
|
|
|
|
2015-11-02 15:38:16 +00:00
|
|
|
flow.handleError(config.flows["t1"].nodes["1"],"my-error",{a:"foo"});
|
2023-06-23 02:11:57 +01:00
|
|
|
await NR_TEST_UTILS.sleep(50)
|
|
|
|
currentNodes["sn"].should.have.a.property("handled",0);
|
|
|
|
currentNodes["sn2"].should.have.a.property("handled",1);
|
|
|
|
var statusMessage = currentNodes["sn2"].messages[0];
|
|
|
|
|
|
|
|
statusMessage.should.have.a.property("error");
|
|
|
|
statusMessage.error.should.have.a.property("message","my-error");
|
|
|
|
statusMessage.error.should.have.a.property("source");
|
|
|
|
statusMessage.error.source.should.have.a.property("id","1");
|
|
|
|
statusMessage.error.source.should.have.a.property("type","test");
|
|
|
|
statusMessage.error.source.should.have.a.property("name","a");
|
|
|
|
|
|
|
|
// Node sn3/4 have uncaught:true - so should not get called
|
|
|
|
currentNodes["sn3"].should.have.a.property("handled",0);
|
|
|
|
currentNodes["sn4"].should.have.a.property("handled",0);
|
|
|
|
|
|
|
|
// Inject error that sn1/2 will ignore - so should get picked up by sn3
|
|
|
|
flow.handleError(config.flows["t1"].nodes["3"],"my-error-2",{a:"foo-2"});
|
|
|
|
|
|
|
|
await NR_TEST_UTILS.sleep(50)
|
|
|
|
currentNodes["sn"].should.have.a.property("handled",0);
|
|
|
|
currentNodes["sn2"].should.have.a.property("handled",1);
|
|
|
|
currentNodes["sn3"].should.have.a.property("handled",1);
|
|
|
|
currentNodes["sn4"].should.have.a.property("handled",1);
|
|
|
|
statusMessage = currentNodes["sn3"].messages[0];
|
|
|
|
statusMessage.should.have.a.property("error");
|
|
|
|
statusMessage.error.should.have.a.property("message","my-error-2");
|
|
|
|
statusMessage.error.should.have.a.property("source");
|
|
|
|
statusMessage.error.source.should.have.a.property("id","3");
|
|
|
|
statusMessage.error.source.should.have.a.property("type","test");
|
|
|
|
await flow.stop()
|
2015-01-16 15:43:47 +00:00
|
|
|
});
|
2023-06-23 02:11:57 +01:00
|
|
|
it("passes an error event to the group scoped catch node",async function() {
|
2023-05-22 22:33:31 +01:00
|
|
|
var config = flowUtils.parseConfig([
|
|
|
|
{id:"t1",type:"tab"},
|
2023-06-23 02:11:57 +01:00
|
|
|
{id: "g1", type: "group", g: "g3", z:"t1" },
|
|
|
|
{id: "g2", type: "group", z:"t1" },
|
|
|
|
{id: "g3", type: "group", z:"t1" },
|
2023-05-22 22:33:31 +01:00
|
|
|
{id:"1",x:10,y:10,z:"t1",g:"g1", type:"test",name:"a",wires:["2"]},
|
|
|
|
// sn - in the same group as source node
|
|
|
|
{id:"sn",x:10,y:10,z:"t1",g:"g1", type:"catch",scope:"group",wires:[]},
|
|
|
|
// sn2 - in a different group hierarchy to the source node
|
|
|
|
{id:"sn2",x:10,y:10,z:"t1", g:"g2", type:"catch",scope:"group",wires:[]},
|
|
|
|
// sn3 - in a higher-level group to the source node
|
|
|
|
{id:"sn3",x:10,y:10,z:"t1", g:"g3", type:"catch",scope:"group",wires:[]},
|
|
|
|
// sn2 - in a different group hierarchy, but not scope to the group
|
|
|
|
{id:"sn4",x:10,y:10,z:"t1", g:"g2", type:"catch",wires:[]},
|
|
|
|
|
|
|
|
]);
|
|
|
|
var flow = Flow.create({},config,config.flows["t1"]);
|
|
|
|
|
2023-06-23 02:11:57 +01:00
|
|
|
await flow.start();
|
2023-05-22 22:33:31 +01:00
|
|
|
|
|
|
|
var activeNodes = flow.getActiveNodes();
|
|
|
|
|
|
|
|
flow.handleError(config.flows["t1"].nodes["1"],"my-error",{a:"foo"});
|
2023-06-23 02:11:57 +01:00
|
|
|
await NR_TEST_UTILS.sleep(50)
|
|
|
|
currentNodes["sn"].should.have.a.property("handled",1);
|
|
|
|
currentNodes["sn2"].should.have.a.property("handled",0);
|
|
|
|
currentNodes["sn3"].should.have.a.property("handled",1);
|
|
|
|
currentNodes["sn3"].should.have.a.property("handled",1);
|
|
|
|
await flow.stop()
|
2023-05-22 22:33:31 +01:00
|
|
|
});
|
2023-06-23 02:11:57 +01:00
|
|
|
|
|
|
|
it("moves any existing error object sideways", async function() {
|
2015-11-02 15:38:16 +00:00
|
|
|
var config = flowUtils.parseConfig([
|
|
|
|
{id:"t1",type:"tab"},
|
|
|
|
{id:"1",x:10,y:10,z:"t1",type:"test",name:"a",wires:["2"]},
|
|
|
|
{id:"2",x:10,y:10,z:"t1",type:"test",wires:["3"]},
|
|
|
|
{id:"3",x:10,y:10,z:"t1",type:"test",foo:"a",wires:[]},
|
|
|
|
{id:"sn",x:10,y:10,z:"t1",type:"catch",foo:"a",wires:[]}
|
|
|
|
]);
|
2019-01-16 16:27:19 +00:00
|
|
|
var flow = Flow.create({},config,config.flows["t1"]);
|
2015-11-02 15:38:16 +00:00
|
|
|
|
2023-06-23 02:11:57 +01:00
|
|
|
await flow.start();
|
2015-11-02 15:38:16 +00:00
|
|
|
|
|
|
|
var activeNodes = flow.getActiveNodes();
|
|
|
|
|
|
|
|
flow.handleError(config.flows["t1"].nodes["1"],"my-error",{a:"foo",error:"existing"});
|
2023-06-23 02:11:57 +01:00
|
|
|
await NR_TEST_UTILS.sleep(50)
|
|
|
|
currentNodes["sn"].should.have.a.property("handled",1);
|
|
|
|
var statusMessage = currentNodes["sn"].messages[0];
|
|
|
|
|
|
|
|
statusMessage.should.have.a.property("_error","existing");
|
|
|
|
statusMessage.should.have.a.property("error");
|
|
|
|
statusMessage.error.should.have.a.property("message","my-error");
|
|
|
|
statusMessage.error.should.have.a.property("source");
|
|
|
|
statusMessage.error.source.should.have.a.property("id","1");
|
|
|
|
statusMessage.error.source.should.have.a.property("type","test");
|
|
|
|
statusMessage.error.source.should.have.a.property("name","a");
|
|
|
|
|
|
|
|
await flow.stop()
|
2019-07-08 23:23:33 +01:00
|
|
|
});
|
|
|
|
it("prevents an error looping more than 10 times",function(){});
|
|
|
|
});
|
|
|
|
|
|
|
|
describe("#handleComplete",function() {
|
2023-06-23 02:11:57 +01:00
|
|
|
it("passes a complete event to the adjacent Complete node",async function() {
|
2019-07-08 23:23:33 +01:00
|
|
|
var config = flowUtils.parseConfig([
|
|
|
|
{id:"t1",type:"tab"},
|
|
|
|
{id:"1",x:10,y:10,z:"t1",type:"testDone",name:"a",wires:["2"]},
|
|
|
|
{id:"2",x:10,y:10,z:"t1",type:"test",wires:["3"]},
|
|
|
|
{id:"3",x:10,y:10,z:"t1",type:"testDone",foo:"a",wires:[]},
|
|
|
|
{id:"cn",x:10,y:10,z:"t1",type:"complete",scope:["1","3"],foo:"a",wires:[]}
|
|
|
|
]);
|
|
|
|
var flow = Flow.create({},config,config.flows["t1"]);
|
2015-11-02 15:38:16 +00:00
|
|
|
|
2023-06-23 02:11:57 +01:00
|
|
|
await flow.start();
|
2015-11-02 15:38:16 +00:00
|
|
|
|
2019-07-08 23:23:33 +01:00
|
|
|
var activeNodes = flow.getActiveNodes();
|
|
|
|
Object.keys(activeNodes).should.have.length(4);
|
2015-11-02 15:38:16 +00:00
|
|
|
|
2019-07-08 23:23:33 +01:00
|
|
|
var msg = {payload: "hello world"}
|
|
|
|
var n1 = currentNodes["1"].receive(msg);
|
2023-06-23 02:11:57 +01:00
|
|
|
await NR_TEST_UTILS.sleep(50)
|
|
|
|
|
|
|
|
currentNodes["cn"].should.have.a.property("handled",2);
|
|
|
|
currentNodes["cn"].messages[0].should.have.a.property("handled",1);
|
|
|
|
currentNodes["cn"].messages[1].should.have.a.property("handled",2);
|
|
|
|
await flow.stop()
|
2015-02-22 23:15:55 +00:00
|
|
|
});
|
2015-01-15 17:10:32 +00:00
|
|
|
});
|
2019-07-08 23:23:33 +01:00
|
|
|
|
|
|
|
|
2020-07-30 17:52:39 +01:00
|
|
|
describe("#send", function() {
|
2023-06-23 02:11:57 +01:00
|
|
|
it("sends a message - no cloning", async function() {
|
2020-07-30 17:52:39 +01:00
|
|
|
var config = flowUtils.parseConfig([
|
|
|
|
{id:"t1",type:"tab"},
|
|
|
|
{id:"1",x:10,y:10,z:"t1",type:"test",foo:"a",wires:["2"]},
|
|
|
|
{id:"2",x:10,y:10,z:"t1",type:"test",foo:"a",wires:["3"]}
|
|
|
|
]);
|
|
|
|
var flow = Flow.create({},config,config.flows["t1"]);
|
2023-06-23 02:11:57 +01:00
|
|
|
await flow.start();
|
2020-07-30 17:52:39 +01:00
|
|
|
|
|
|
|
Object.keys(flow.getActiveNodes()).should.have.length(2);
|
|
|
|
|
|
|
|
var n1 = flow.getNode('1');
|
|
|
|
var n2 = flow.getNode('2');
|
|
|
|
var messageReceived = false;
|
2023-06-23 02:11:57 +01:00
|
|
|
|
|
|
|
return new Promise((resolve, reject) => {
|
|
|
|
const shutdownTest = async function(err) {
|
|
|
|
hooks.clear();
|
|
|
|
await flow.stop()
|
|
|
|
if (err) { reject(err) }
|
|
|
|
else { resolve() }
|
|
|
|
}
|
|
|
|
n2.receive = function(msg) {
|
|
|
|
messageReceived = true;
|
|
|
|
try {
|
|
|
|
msg.should.be.exactly(message);
|
|
|
|
shutdownTest();
|
|
|
|
} catch(err) {
|
|
|
|
shutdownTest(err);
|
|
|
|
}
|
2020-07-30 17:52:39 +01:00
|
|
|
}
|
|
|
|
|
2023-06-23 02:11:57 +01:00
|
|
|
var message = {payload:"hello"}
|
|
|
|
flow.send([{
|
|
|
|
msg: message,
|
|
|
|
source: { id:"1", node: n1 },
|
|
|
|
destination: { id:"2", node: undefined },
|
|
|
|
cloneMessage: false
|
|
|
|
}])
|
|
|
|
messageReceived.should.be.false()
|
|
|
|
})
|
2020-07-30 17:52:39 +01:00
|
|
|
})
|
2023-06-23 02:11:57 +01:00
|
|
|
it("sends a message - cloning", async function() {
|
2020-07-30 17:52:39 +01:00
|
|
|
var config = flowUtils.parseConfig([
|
|
|
|
{id:"t1",type:"tab"},
|
|
|
|
{id:"1",x:10,y:10,z:"t1",type:"test",foo:"a",wires:["2"]},
|
|
|
|
{id:"2",x:10,y:10,z:"t1",type:"test",foo:"a",wires:["3"]}
|
|
|
|
]);
|
|
|
|
var flow = Flow.create({},config,config.flows["t1"]);
|
2023-06-23 02:11:57 +01:00
|
|
|
await flow.start();
|
2020-07-30 17:52:39 +01:00
|
|
|
|
|
|
|
Object.keys(flow.getActiveNodes()).should.have.length(2);
|
|
|
|
|
|
|
|
var n1 = flow.getNode('1');
|
|
|
|
var n2 = flow.getNode('2');
|
|
|
|
|
2023-06-23 02:11:57 +01:00
|
|
|
return new Promise((resolve, reject) => {
|
|
|
|
const shutdownTest = async function(err) {
|
|
|
|
hooks.clear();
|
|
|
|
await flow.stop()
|
|
|
|
if (err) { reject(err) }
|
|
|
|
else { resolve() }
|
|
|
|
}
|
|
|
|
n2.receive = function(msg) {
|
|
|
|
try {
|
|
|
|
// Message should be cloned
|
|
|
|
msg.should.be.eql(message);
|
|
|
|
msg.should.not.be.exactly(message);
|
|
|
|
shutdownTest();
|
|
|
|
} catch(err) {
|
|
|
|
shutdownTest(err);
|
|
|
|
}
|
2020-07-30 17:52:39 +01:00
|
|
|
}
|
|
|
|
|
2023-06-23 02:11:57 +01:00
|
|
|
var message = {payload:"hello"}
|
|
|
|
flow.send([{
|
|
|
|
msg: message,
|
|
|
|
source: { id:"1", node: n1 },
|
|
|
|
destination: { id:"2", node: undefined },
|
|
|
|
cloneMessage: true
|
|
|
|
}])
|
|
|
|
})
|
2020-07-30 17:52:39 +01:00
|
|
|
})
|
2023-06-23 02:11:57 +01:00
|
|
|
it("sends multiple messages", async function() {
|
2020-07-30 17:52:39 +01:00
|
|
|
var config = flowUtils.parseConfig([
|
|
|
|
{id:"t1",type:"tab"},
|
|
|
|
{id:"1",x:10,y:10,z:"t1",type:"test",foo:"a",wires:["2"]},
|
|
|
|
{id:"2",x:10,y:10,z:"t1",type:"test",foo:"a",wires:["3"]}
|
|
|
|
]);
|
|
|
|
var flow = Flow.create({},config,config.flows["t1"]);
|
2023-06-23 02:11:57 +01:00
|
|
|
await flow.start();
|
2020-07-30 17:52:39 +01:00
|
|
|
|
|
|
|
Object.keys(flow.getActiveNodes()).should.have.length(2);
|
|
|
|
|
|
|
|
var n1 = flow.getNode('1');
|
|
|
|
var n2 = flow.getNode('2');
|
2023-06-23 02:11:57 +01:00
|
|
|
return new Promise((resolve, reject) => {
|
|
|
|
const shutdownTest = async function(err) {
|
|
|
|
hooks.clear();
|
|
|
|
await flow.stop()
|
|
|
|
if (err) { reject(err) }
|
|
|
|
else { resolve() }
|
|
|
|
}
|
|
|
|
var messageCount = 0;
|
|
|
|
n2.receive = function(msg) {
|
|
|
|
try {
|
|
|
|
msg.should.be.exactly(messages[messageCount++]);
|
|
|
|
if (messageCount === 2) {
|
|
|
|
shutdownTest();
|
|
|
|
}
|
|
|
|
} catch(err) {
|
|
|
|
shutdownTest(err);
|
2020-07-30 17:52:39 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-06-23 02:11:57 +01:00
|
|
|
var messages = [{payload:"hello"},{payload:"world"}];
|
|
|
|
|
|
|
|
flow.send([{
|
|
|
|
msg: messages[0],
|
|
|
|
source: { id:"1", node: n1 },
|
|
|
|
destination: { id:"2", node: undefined }
|
|
|
|
},{
|
|
|
|
msg: messages[1],
|
|
|
|
source: { id:"1", node: n1 },
|
|
|
|
destination: { id:"2", node: undefined }
|
|
|
|
}])
|
|
|
|
})
|
2020-07-30 17:52:39 +01:00
|
|
|
})
|
2023-06-23 02:11:57 +01:00
|
|
|
it("sends a message - triggers hooks", async function() {
|
|
|
|
const message = {payload:"hello"}
|
2020-07-30 17:52:39 +01:00
|
|
|
var hookErrors = [];
|
|
|
|
var messageReceived = false;
|
|
|
|
var hooksCalled = [];
|
|
|
|
hooks.add("onSend", function(sendEvents) {
|
|
|
|
hooksCalled.push("onSend")
|
|
|
|
try {
|
|
|
|
messageReceived.should.be.false()
|
|
|
|
sendEvents.should.have.length(1);
|
|
|
|
sendEvents[0].msg.should.be.exactly(message);
|
|
|
|
} catch(err) {
|
|
|
|
hookErrors.push(err);
|
|
|
|
}
|
|
|
|
})
|
|
|
|
hooks.add("preRoute", function(sendEvent) {
|
|
|
|
hooksCalled.push("preRoute")
|
|
|
|
try {
|
|
|
|
messageReceived.should.be.false()
|
|
|
|
sendEvent.msg.should.be.exactly(message);
|
|
|
|
should.not.exist(sendEvent.destination.node)
|
|
|
|
} catch(err) {
|
|
|
|
hookErrors.push(err);
|
|
|
|
}
|
|
|
|
|
|
|
|
})
|
|
|
|
hooks.add("preDeliver", function(sendEvent) {
|
|
|
|
hooksCalled.push("preDeliver")
|
|
|
|
try {
|
|
|
|
messageReceived.should.be.false()
|
|
|
|
// Cloning should have happened
|
|
|
|
sendEvent.msg.should.not.be.exactly(message);
|
|
|
|
// Destinatino node populated
|
|
|
|
should.exist(sendEvent.destination.node)
|
|
|
|
} catch(err) {
|
|
|
|
hookErrors.push(err);
|
|
|
|
}
|
|
|
|
|
|
|
|
})
|
|
|
|
hooks.add("postDeliver", function(sendEvent) {
|
|
|
|
hooksCalled.push("postDeliver")
|
|
|
|
try {
|
|
|
|
messageReceived.should.be.false()
|
|
|
|
|
|
|
|
} catch(err) {
|
|
|
|
hookErrors.push(err);
|
|
|
|
}
|
|
|
|
|
|
|
|
})
|
|
|
|
var config = flowUtils.parseConfig([
|
|
|
|
{id:"t1",type:"tab"},
|
|
|
|
{id:"1",x:10,y:10,z:"t1",type:"test",foo:"a",wires:["2"]},
|
|
|
|
{id:"2",x:10,y:10,z:"t1",type:"test",foo:"a",wires:["3"]}
|
|
|
|
]);
|
|
|
|
var flow = Flow.create({},config,config.flows["t1"]);
|
2023-06-23 02:11:57 +01:00
|
|
|
await flow.start();
|
2020-07-30 17:52:39 +01:00
|
|
|
|
|
|
|
Object.keys(flow.getActiveNodes()).should.have.length(2);
|
|
|
|
|
|
|
|
var n1 = flow.getNode('1');
|
|
|
|
var n2 = flow.getNode('2');
|
2023-06-23 02:11:57 +01:00
|
|
|
return new Promise((resolve, reject) => {
|
|
|
|
const shutdownTest = async function(err) {
|
|
|
|
hooks.clear();
|
|
|
|
await flow.stop()
|
|
|
|
if (err) { reject(err) }
|
|
|
|
else { resolve() }
|
|
|
|
}
|
|
|
|
n2.receive = function(msg) {
|
|
|
|
messageReceived = true;
|
|
|
|
try {
|
|
|
|
msg.should.be.eql(message);
|
|
|
|
msg.should.not.be.exactly(message);
|
|
|
|
hooksCalled.should.eql(["onSend","preRoute","preDeliver","postDeliver"])
|
|
|
|
if (hookErrors.length > 0) {
|
|
|
|
shutdownTest(hookErrors[0])
|
|
|
|
} else {
|
|
|
|
shutdownTest();
|
|
|
|
}
|
|
|
|
} catch(err) {
|
|
|
|
shutdownTest(err);
|
2020-07-30 17:52:39 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-06-23 02:11:57 +01:00
|
|
|
|
|
|
|
flow.send([{
|
|
|
|
msg: message,
|
|
|
|
source: { id:"1", node: n1 },
|
|
|
|
destination: { id:"2", node: undefined },
|
|
|
|
cloneMessage: true
|
|
|
|
}])
|
|
|
|
})
|
2020-07-30 17:52:39 +01:00
|
|
|
})
|
|
|
|
|
|
|
|
describe("errors thrown by hooks are reported to the sending node", function() {
|
|
|
|
var flow;
|
|
|
|
var n1,n2;
|
|
|
|
var messageReceived = false;
|
|
|
|
var errorReceived = null;
|
2023-06-23 02:11:57 +01:00
|
|
|
before(async function() {
|
2020-07-30 17:52:39 +01:00
|
|
|
hooks.add("onSend", function(sendEvents) {
|
|
|
|
if (sendEvents[0].msg.payload === "trigger-onSend") {
|
|
|
|
throw new Error("onSend Error");
|
|
|
|
}
|
|
|
|
})
|
|
|
|
hooks.add("preRoute", function(sendEvent) {
|
|
|
|
if (sendEvent.msg.payload === "trigger-preRoute") {
|
|
|
|
throw new Error("preRoute Error");
|
|
|
|
}
|
|
|
|
})
|
|
|
|
hooks.add("preDeliver", function(sendEvent) {
|
|
|
|
if (sendEvent.msg.payload === "trigger-preDeliver") {
|
|
|
|
throw new Error("preDeliver Error");
|
|
|
|
}
|
|
|
|
})
|
|
|
|
hooks.add("postDeliver", function(sendEvent) {
|
|
|
|
if (sendEvent.msg.payload === "trigger-postDeliver") {
|
|
|
|
throw new Error("postDeliver Error");
|
|
|
|
}
|
|
|
|
})
|
|
|
|
var config = flowUtils.parseConfig([
|
|
|
|
{id:"t1",type:"tab"},
|
|
|
|
{id:"1",x:10,y:10,z:"t1",type:"test",foo:"a",wires:["2"]},
|
|
|
|
{id:"2",x:10,y:10,z:"t1",type:"test",foo:"a",wires:["3"]}
|
|
|
|
]);
|
|
|
|
flow = Flow.create({},config,config.flows["t1"]);
|
2023-06-23 02:11:57 +01:00
|
|
|
await flow.start();
|
2020-07-30 17:52:39 +01:00
|
|
|
n1 = flow.getNode('1');
|
|
|
|
n2 = flow.getNode('2');
|
|
|
|
n2.receive = function(msg) {
|
|
|
|
messageReceived = true;
|
|
|
|
}
|
|
|
|
n1.error = function(err) {
|
|
|
|
errorReceived = err;
|
|
|
|
}
|
|
|
|
|
|
|
|
})
|
2023-06-23 02:11:57 +01:00
|
|
|
after(async function() {
|
2020-07-30 17:52:39 +01:00
|
|
|
hooks.clear();
|
2023-06-23 02:11:57 +01:00
|
|
|
await flow.stop()
|
2020-07-30 17:52:39 +01:00
|
|
|
})
|
|
|
|
beforeEach(function() {
|
|
|
|
messageReceived = false;
|
|
|
|
errorReceived = null;
|
|
|
|
})
|
|
|
|
function testHook(hook, msgExpected, done) {
|
|
|
|
var message = {payload:"trigger-"+hook}
|
|
|
|
flow.send([{
|
|
|
|
msg: message,
|
|
|
|
source: { id:"1", node: n1 },
|
|
|
|
destination: { id:"2", node: undefined },
|
|
|
|
cloneMessage: true
|
|
|
|
}])
|
|
|
|
setTimeout(function() {
|
|
|
|
try {
|
|
|
|
messageReceived.should.equal(msgExpected);
|
|
|
|
should.exist(errorReceived)
|
|
|
|
errorReceived.toString().should.containEql(hook);
|
|
|
|
done();
|
|
|
|
} catch(err) {
|
|
|
|
done(err);
|
|
|
|
}
|
|
|
|
},10)
|
|
|
|
}
|
|
|
|
|
|
|
|
it("onSend", function(done) { testHook("onSend", false, done) })
|
|
|
|
it("preRoute", function(done) { testHook("preRoute", false, done) })
|
|
|
|
it("preDeliver", function(done) { testHook("preDeliver", false, done) })
|
|
|
|
it("postDeliver", function(done) { testHook("postDeliver", true, done) })
|
|
|
|
})
|
|
|
|
|
|
|
|
describe("hooks can stop the sending of messages", function() {
|
|
|
|
var flow;
|
|
|
|
var n1,n2;
|
|
|
|
var messageReceived = false;
|
|
|
|
var errorReceived = false;
|
2023-06-23 02:11:57 +01:00
|
|
|
before(async function() {
|
2020-07-30 17:52:39 +01:00
|
|
|
hooks.add("onSend", function(sendEvents) {
|
|
|
|
if (sendEvents[0].msg.payload === "trigger-onSend") {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
})
|
|
|
|
hooks.add("preRoute", function(sendEvent) {
|
|
|
|
if (sendEvent.msg.payload === "trigger-preRoute") {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
})
|
|
|
|
hooks.add("preDeliver", function(sendEvent) {
|
|
|
|
if (sendEvent.msg.payload === "trigger-preDeliver") {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
})
|
|
|
|
hooks.add("postDeliver", function(sendEvent) {
|
|
|
|
if (sendEvent.msg.payload === "trigger-postDeliver") {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
})
|
|
|
|
var config = flowUtils.parseConfig([
|
|
|
|
{id:"t1",type:"tab"},
|
|
|
|
{id:"1",x:10,y:10,z:"t1",type:"test",foo:"a",wires:["2"]},
|
|
|
|
{id:"2",x:10,y:10,z:"t1",type:"test",foo:"a",wires:["3"]}
|
|
|
|
]);
|
|
|
|
flow = Flow.create({},config,config.flows["t1"]);
|
2023-06-23 02:11:57 +01:00
|
|
|
await flow.start();
|
2020-07-30 17:52:39 +01:00
|
|
|
n1 = flow.getNode('1');
|
|
|
|
n2 = flow.getNode('2');
|
|
|
|
n2.receive = function(msg) {
|
|
|
|
messageReceived = true;
|
|
|
|
}
|
|
|
|
n1.error = function(err) {
|
|
|
|
errorReceived = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
})
|
2023-06-23 02:11:57 +01:00
|
|
|
after(async function() {
|
2020-07-30 17:52:39 +01:00
|
|
|
hooks.clear();
|
2023-06-23 02:11:57 +01:00
|
|
|
await flow.stop()
|
2020-07-30 17:52:39 +01:00
|
|
|
})
|
|
|
|
function testSend(payload,messageReceivedExpected,errorReceivedExpected,done) {
|
|
|
|
messageReceived = false;
|
|
|
|
errorReceived = false;
|
|
|
|
flow.send([{
|
|
|
|
msg: {payload: payload},
|
|
|
|
source: { id:"1", node: n1 },
|
|
|
|
destination: { id:"2", node: undefined },
|
|
|
|
cloneMessage: true
|
|
|
|
}])
|
|
|
|
setTimeout(function() {
|
|
|
|
try {
|
|
|
|
messageReceived.should.eql(messageReceivedExpected)
|
|
|
|
errorReceived.should.eql(errorReceivedExpected)
|
|
|
|
done();
|
|
|
|
} catch(err) {
|
|
|
|
done(err);
|
|
|
|
}
|
|
|
|
},10)
|
|
|
|
}
|
|
|
|
function testHook(hook, done) {
|
|
|
|
testSend("pass",true,false,err => {
|
|
|
|
if (err) {
|
|
|
|
done(err)
|
|
|
|
} else {
|
|
|
|
testSend("trigger-"+hook,false,false,done);
|
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
it("onSend", function(done) { testHook("onSend", done) })
|
|
|
|
it("preRoute", function(done) { testHook("preRoute", done) })
|
|
|
|
it("preDeliver", function(done) { testHook("preDeliver", done) })
|
|
|
|
// postDeliver happens after delivery is scheduled so cannot stop it
|
|
|
|
// it("postDeliver", function(done) { testHook("postDeliver", done) })
|
|
|
|
})
|
|
|
|
})
|
|
|
|
|
2021-08-19 21:15:13 +09:00
|
|
|
describe("#env", function () {
|
2023-06-23 02:11:57 +01:00
|
|
|
it("can instantiate a node with environment variable property values of group and tab", async function () {
|
|
|
|
after(function() {
|
|
|
|
delete process.env.V0;
|
|
|
|
delete process.env.V1;
|
|
|
|
})
|
|
|
|
process.env.V0 = "gv0";
|
|
|
|
process.env.V1 = "gv1";
|
|
|
|
process.env.V3 = "gv3";
|
|
|
|
var config = flowUtils.parseConfig([
|
|
|
|
{id:"t1",type:"tab",env:[
|
|
|
|
{"name": "V0", value: "t1v0", type: "str"},
|
|
|
|
{"name": "V2", value: "t1v2", type: "str"}
|
|
|
|
]},
|
|
|
|
{id:"g1",type:"group",z:"t1",env:[
|
|
|
|
{"name": "V0", value: "g1v0", type: "str"},
|
|
|
|
{"name": "V1", value: "g1v1", type: "str"}
|
|
|
|
]},
|
|
|
|
{id:"g2",type:"group",z:"t1",g:"g1",env:[
|
|
|
|
{"name": "V1", value: "g2v1", type: "str"}
|
|
|
|
]},
|
|
|
|
{id:"t1__V0",x:10,y:10,z:"t1",type:"test",foo:"${V0}",wires:[]}, // V0 will come from tab env V0
|
|
|
|
{id:"t1g1V0",x:10,y:10,z:"t1",g:"g1",type:"test",foo:"${V0}",wires:[]}, // V0 will come from group 1 env V0
|
|
|
|
{id:"t1g1V1",x:10,y:10,z:"t1",g:"g1",type:"test",foo:"${V1}",wires:[]}, // V1 will come from group 1 env V1
|
|
|
|
{id:"t1g2V0",x:10,y:10,z:"t1",g:"g2",type:"test",foo:"${V0}",wires:[]}, // V0 will come from group 1 env V0
|
|
|
|
{id:"t1g2V1",x:10,y:10,z:"t1",g:"g2",type:"test",foo:"${V1}",wires:[]}, // V1 will come from group 2 env V1
|
|
|
|
{id:"t1g2V2",x:10,y:10,z:"t1",g:"g2",type:"test",foo:"${V2}",wires:[]}, // V2 will come from tab 1 env V2
|
|
|
|
{id:"t1g2V3",x:10,y:10,z:"t1",g:"g2",type:"test",foo:"${V3}",wires:[]}, // V3 will come from process env V3
|
|
|
|
|
|
|
|
{id:"t1__V1",x:10,y:10,z:"t1",type:"test",foo:"${V1}",wires:[]},
|
|
|
|
]);
|
|
|
|
var flow = Flow.create({getSetting:v=>process.env[v]},config,config.flows["t1"]);
|
|
|
|
await flow.start();
|
|
|
|
|
|
|
|
var activeNodes = flow.getActiveNodes();
|
|
|
|
|
|
|
|
activeNodes.t1__V0.foo.should.equal("t1v0"); // node in tab 1, get tab 1 env V0
|
|
|
|
activeNodes.t1__V1.foo.should.equal("gv1"); // node in tab 1, get V1, (tab 1 no V1) --> parent (global has V1)
|
|
|
|
activeNodes.t1g1V0.foo.should.equal("g1v0"); // node in group 1, get V0, (group 1 has V0)
|
|
|
|
activeNodes.t1g1V1.foo.should.equal("g1v1"); // node in group 1, get V1, (group 1 has V1)
|
|
|
|
activeNodes.t1g2V0.foo.should.equal("g1v0"); // node in group 2, get V0, (group 2 no V0) --> parent (group 1 has V0)
|
|
|
|
activeNodes.t1g2V1.foo.should.equal("g2v1"); // node in group 2, get V1, (group 2 has V1)
|
|
|
|
activeNodes.t1g2V2.foo.should.equal("t1v2"); // node in group 2, get V2, (group 2 no V2) --> parent (tab 1 has V2)
|
|
|
|
activeNodes.t1g2V3.foo.should.equal("gv3"); // node in group 2, get V3, (group 2 no V3) --> parent (tab 1 no V2) --> parent (global has V3)
|
|
|
|
|
|
|
|
await flow.stop()
|
2021-08-19 21:15:13 +09:00
|
|
|
});
|
2021-11-27 19:29:57 +09:00
|
|
|
|
2023-06-23 02:11:57 +01:00
|
|
|
it("can access environment variable property using $parent", async function () {
|
|
|
|
after(function() {
|
|
|
|
delete process.env.V0;
|
|
|
|
delete process.env.V1;
|
|
|
|
})
|
|
|
|
process.env.V0 = "gv0";
|
|
|
|
process.env.V1 = "gv1";
|
|
|
|
var config = flowUtils.parseConfig([
|
|
|
|
{id:"t1",type:"tab",env:[
|
|
|
|
{"name": "V0", value: "v0", type: "str"}
|
|
|
|
]},
|
|
|
|
{id:"g1",type:"group",z:"t1",env:[
|
|
|
|
{"name": "V0", value: "v1", type: "str"},
|
|
|
|
{"name": "V1", value: "v2", type: "str"}
|
|
|
|
]},
|
|
|
|
{id:"g2",type:"group",z:"t1",g:"g1",env:[
|
|
|
|
{"name": "V1", value: "v3", type: "str"}
|
|
|
|
]},
|
|
|
|
{id:"1",x:10,y:10,z:"t1",type:"test",foo:"${$parent.V0}",wires:[]},
|
|
|
|
{id:"2",x:10,y:10,z:"t1",g:"g1",type:"test",foo:"${$parent.V0}",wires:[]},
|
|
|
|
{id:"3",x:10,y:10,z:"t1",g:"g1",type:"test",foo:"${$parent.V1}",wires:[]},
|
|
|
|
{id:"4",x:10,y:10,z:"t1",g:"g2",type:"test",foo:"${$parent.V1}",wires:[]},
|
|
|
|
{id:"5",x:10,y:10,z:"t1",type:"test",foo:"${$parent.V1}",wires:[]},
|
|
|
|
]);
|
|
|
|
var flow = Flow.create({getSetting:v=>process.env[v]},config,config.flows["t1"]);
|
|
|
|
await flow.start();
|
|
|
|
|
|
|
|
var activeNodes = flow.getActiveNodes();
|
2021-11-27 19:29:57 +09:00
|
|
|
|
2023-06-23 02:11:57 +01:00
|
|
|
activeNodes["1"].foo.should.equal("gv0");
|
|
|
|
activeNodes["2"].foo.should.equal("v0");
|
|
|
|
activeNodes["3"].foo.should.equal("gv1");
|
|
|
|
activeNodes["4"].foo.should.equal("v2");
|
|
|
|
activeNodes["5"].foo.should.equal("gv1");
|
2021-11-27 19:29:57 +09:00
|
|
|
|
2023-06-23 02:11:57 +01:00
|
|
|
await flow.stop()
|
2021-11-27 19:29:57 +09:00
|
|
|
});
|
2021-08-19 21:15:13 +09:00
|
|
|
|
2023-06-23 02:11:57 +01:00
|
|
|
it("can define environment variable using JSONata", async function () {
|
|
|
|
after(function() {
|
|
|
|
delete process.env.V0;
|
|
|
|
})
|
|
|
|
var config = flowUtils.parseConfig([
|
|
|
|
{id:"t1",type:"tab",env:[
|
|
|
|
{"name": "V0", value: "1+2", type: "jsonata"}
|
|
|
|
]},
|
|
|
|
{id:"g1",type:"group",z:"t1",env:[
|
|
|
|
{"name": "V1", value: "2+3", type: "jsonata"},
|
|
|
|
]},
|
|
|
|
{id:"1",x:10,y:10,z:"t1",g:"g1",type:"test",foo:"$(V0)",wires:[]},
|
|
|
|
{id:"2",x:10,y:10,z:"t1",g:"g1",type:"test",foo:"$(V1)",wires:[]},
|
|
|
|
]);
|
|
|
|
var flow = Flow.create({getSetting:v=>process.env[v]},config,config.flows["t1"]);
|
|
|
|
await flow.start();
|
2022-07-27 20:25:43 +09:00
|
|
|
|
2023-06-23 02:11:57 +01:00
|
|
|
var activeNodes = flow.getActiveNodes();
|
2022-07-27 20:25:43 +09:00
|
|
|
|
2023-06-23 02:11:57 +01:00
|
|
|
activeNodes["1"].foo.should.equal(3);
|
|
|
|
activeNodes["2"].foo.should.equal(5);
|
2022-07-27 20:25:43 +09:00
|
|
|
|
2023-06-23 02:11:57 +01:00
|
|
|
await flow.stop()
|
2022-07-27 20:25:43 +09:00
|
|
|
});
|
|
|
|
|
2023-06-23 02:11:57 +01:00
|
|
|
it("can access global environment variables defined as JSONata values", async function () {
|
|
|
|
after(function() {
|
|
|
|
delete process.env.V0;
|
|
|
|
})
|
|
|
|
var config = flowUtils.parseConfig([
|
|
|
|
{id:"t1",type:"tab",env:[
|
|
|
|
{"name": "V0", value: "1+2", type: "jsonata"}
|
|
|
|
]},
|
|
|
|
{id:"g1",type:"group",z:"t1",env:[
|
|
|
|
{"name": "V1", value: "2+3", type: "jsonata"},
|
|
|
|
]},
|
|
|
|
{id:"1",x:10,y:10,z:"t1",g:"g1",type:"test",foo:"$(V0)",wires:[]},
|
|
|
|
{id:"2",x:10,y:10,z:"t1",g:"g1",type:"test",foo:"$(V1)",wires:[]},
|
|
|
|
]);
|
|
|
|
var flow = Flow.create({getSetting:v=>process.env[v]},config,config.flows["t1"]);
|
|
|
|
await flow.start();
|
2023-06-22 10:24:29 +01:00
|
|
|
|
2023-06-23 02:11:57 +01:00
|
|
|
var activeNodes = flow.getActiveNodes();
|
2023-06-22 10:24:29 +01:00
|
|
|
|
2023-06-23 02:11:57 +01:00
|
|
|
activeNodes["1"].foo.should.equal(3);
|
|
|
|
activeNodes["2"].foo.should.equal(5);
|
2023-06-22 10:24:29 +01:00
|
|
|
|
2023-06-23 02:11:57 +01:00
|
|
|
await flow.stop()
|
2023-06-22 10:24:29 +01:00
|
|
|
});
|
2023-06-23 09:35:00 +01:00
|
|
|
it("global flow can access global-config defined environment variables", async function () {
|
|
|
|
after(function() {
|
|
|
|
delete process.env.V0;
|
|
|
|
})
|
|
|
|
const config = flowUtils.parseConfig([
|
|
|
|
{id:"gc", type:"global-config", env:[
|
|
|
|
{"name": "GC0", value: "3+4", type: "jsonata"}
|
|
|
|
]},
|
|
|
|
{id:"t1",type:"tab" },
|
|
|
|
{id:"1",x:10,y:10,z:"t1",type:"test",foo:"${GC0}",wires:[]},
|
|
|
|
]);
|
|
|
|
// Two-arg call - makes this the global flow that handles global-config nodes
|
|
|
|
const globalFlow = Flow.create({getSetting:v=>process.env[v]},config);
|
|
|
|
await globalFlow.start();
|
|
|
|
|
|
|
|
// Pass the globalFlow in as the parent flow to allow global-config lookup
|
|
|
|
const flow = Flow.create(globalFlow,config,config.flows["t1"]);
|
|
|
|
await flow.start();
|
|
|
|
|
|
|
|
var activeNodes = flow.getActiveNodes();
|
|
|
|
activeNodes["1"].foo.should.equal(7);
|
|
|
|
|
|
|
|
await flow.stop()
|
|
|
|
await globalFlow.stop()
|
|
|
|
});
|
2021-08-19 21:15:13 +09:00
|
|
|
});
|
|
|
|
|
2015-10-11 20:37:11 +01:00
|
|
|
});
|