2015-10-11 20:37:11 +01:00
|
|
|
/**
|
2017-01-11 15:24:33 +00:00
|
|
|
* Copyright JS Foundation and other contributors, http://js.foundation
|
2015-10-11 20:37:11 +01: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.
|
|
|
|
**/
|
|
|
|
|
|
|
|
var should = require("should");
|
|
|
|
var sinon = require("sinon");
|
|
|
|
var clone = require("clone");
|
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 flows = NR_TEST_UTILS.require("@node-red/runtime/lib/flows");
|
2018-08-20 16:17:24 +01:00
|
|
|
var RedNode = NR_TEST_UTILS.require("@node-red/runtime/lib/nodes/Node");
|
|
|
|
var RED = NR_TEST_UTILS.require("@node-red/runtime/lib/nodes");
|
2020-12-02 09:25:10 +00:00
|
|
|
var events = NR_TEST_UTILS.require("@node-red/util/lib/events");
|
2018-08-20 16:17:24 +01:00
|
|
|
var credentials = NR_TEST_UTILS.require("@node-red/runtime/lib/nodes/credentials");
|
|
|
|
var typeRegistry = NR_TEST_UTILS.require("@node-red/registry")
|
2020-07-20 16:48:47 +01:00
|
|
|
var Flow = NR_TEST_UTILS.require("@node-red/runtime/lib/flows/Flow");
|
2015-10-11 20:37:11 +01:00
|
|
|
|
|
|
|
describe('flows/index', function() {
|
|
|
|
|
|
|
|
var storage;
|
2015-11-02 15:38:16 +00:00
|
|
|
var eventsOn;
|
2015-10-11 20:37:11 +01:00
|
|
|
var credentialsClean;
|
|
|
|
var credentialsLoad;
|
2021-03-06 20:27:51 +00:00
|
|
|
var credentialsAdd;
|
2015-10-11 20:37:11 +01:00
|
|
|
|
|
|
|
var flowCreate;
|
|
|
|
var getType;
|
2021-02-13 00:18:04 +00:00
|
|
|
var checkFlowDependencies;
|
2015-10-11 20:37:11 +01:00
|
|
|
|
2018-04-23 14:24:51 +01:00
|
|
|
var mockLog = {
|
|
|
|
log: sinon.stub(),
|
|
|
|
debug: sinon.stub(),
|
|
|
|
trace: sinon.stub(),
|
|
|
|
warn: sinon.stub(),
|
|
|
|
info: sinon.stub(),
|
|
|
|
metric: sinon.stub(),
|
|
|
|
_: function() { return "abc"}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2015-10-11 20:37:11 +01: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
|
|
|
return type.indexOf('missing') === -1;
|
2015-10-11 20:37:11 +01:00
|
|
|
});
|
2021-04-09 11:22:57 +01:00
|
|
|
checkFlowDependencies = sinon.stub(typeRegistry, "checkFlowDependencies").callsFake(async function(flow) {
|
2021-02-13 00:18:04 +00:00
|
|
|
if (flow[0].id === "node-with-missing-modules") {
|
|
|
|
throw new Error("Missing module");
|
|
|
|
}
|
|
|
|
});
|
2015-10-11 20:37:11 +01:00
|
|
|
});
|
2021-02-13 00:18:04 +00:00
|
|
|
|
2015-10-11 20:37:11 +01:00
|
|
|
after(function() {
|
|
|
|
getType.restore();
|
2021-02-13 00:18:04 +00:00
|
|
|
checkFlowDependencies.restore();
|
2015-10-11 20:37:11 +01:00
|
|
|
});
|
|
|
|
|
|
|
|
|
|
|
|
beforeEach(function() {
|
2015-11-02 15:38:16 +00:00
|
|
|
eventsOn = sinon.spy(events,"on");
|
2021-04-09 11:22:57 +01:00
|
|
|
credentialsClean = sinon.stub(credentials,"clean").callsFake(function(conf) {
|
2016-09-21 21:58:50 +01:00
|
|
|
conf.forEach(function(n) {
|
|
|
|
delete n.credentials;
|
|
|
|
});
|
2020-11-30 14:38:48 +00:00
|
|
|
return Promise.resolve();
|
2015-10-11 20:37:11 +01:00
|
|
|
});
|
2021-04-09 11:22:57 +01:00
|
|
|
credentialsLoad = sinon.stub(credentials,"load").callsFake(function(creds) {
|
2020-02-13 16:44:48 +00:00
|
|
|
if (creds && creds.hasOwnProperty("$") && creds['$'] === "fail") {
|
2020-11-30 14:38:48 +00:00
|
|
|
return Promise.reject("creds error");
|
2020-02-13 16:44:48 +00:00
|
|
|
}
|
2020-11-30 14:38:48 +00:00
|
|
|
return Promise.resolve();
|
2015-10-11 20:37:11 +01:00
|
|
|
});
|
2021-04-09 11:22:57 +01:00
|
|
|
credentialsAdd = sinon.stub(credentials,"add").callsFake(async function(id, conf){})
|
|
|
|
flowCreate = sinon.stub(Flow,"create").callsFake(function(parent, global, flow) {
|
2015-10-11 20:37:11 +01:00
|
|
|
var id;
|
|
|
|
if (typeof flow === 'undefined') {
|
|
|
|
flow = global;
|
|
|
|
id = '_GLOBAL_';
|
|
|
|
} else {
|
|
|
|
id = flow.id;
|
|
|
|
}
|
|
|
|
flowCreate.flows[id] = {
|
2015-11-02 15:38:16 +00:00
|
|
|
flow: flow,
|
|
|
|
global: global,
|
2015-10-11 20:37:11 +01:00
|
|
|
start: sinon.spy(),
|
|
|
|
update: sinon.spy(),
|
2015-11-02 15:38:16 +00:00
|
|
|
stop: sinon.spy(),
|
|
|
|
getActiveNodes: function() {
|
|
|
|
return flow.nodes||{};
|
|
|
|
},
|
|
|
|
handleError: sinon.spy(),
|
|
|
|
handleStatus: sinon.spy()
|
|
|
|
|
2015-10-11 20:37:11 +01:00
|
|
|
}
|
|
|
|
return flowCreate.flows[id];
|
|
|
|
});
|
|
|
|
flowCreate.flows = {};
|
|
|
|
|
|
|
|
storage = {
|
|
|
|
saveFlows: function(conf) {
|
|
|
|
storage.conf = conf;
|
2020-11-30 14:38:48 +00:00
|
|
|
return Promise.resolve();
|
2015-10-11 20:37:11 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
2015-11-02 15:38:16 +00:00
|
|
|
afterEach(function(done) {
|
2015-10-11 20:37:11 +01:00
|
|
|
eventsOn.restore();
|
|
|
|
credentialsClean.restore();
|
|
|
|
credentialsLoad.restore();
|
2021-03-06 20:27:51 +00:00
|
|
|
credentialsAdd.restore();
|
2015-10-11 20:37:11 +01:00
|
|
|
flowCreate.restore();
|
2015-11-02 15:38:16 +00:00
|
|
|
|
|
|
|
flows.stopFlows().then(done);
|
|
|
|
|
2015-10-11 20:37:11 +01:00
|
|
|
});
|
2015-11-02 15:38:16 +00:00
|
|
|
// describe('#init',function() {
|
|
|
|
// it('registers the type-registered handler', function() {
|
|
|
|
// flows.init({},{});
|
2016-10-10 13:27:43 +01:00
|
|
|
// eventsOn.calledOnce.should.be.true();
|
2015-11-02 15:38:16 +00:00
|
|
|
// });
|
|
|
|
// });
|
2015-10-11 20:37:11 +01:00
|
|
|
|
|
|
|
describe('#setFlows', function() {
|
|
|
|
it('sets the full flow', function(done) {
|
|
|
|
var originalConfig = [
|
|
|
|
{id:"t1-1",x:10,y:10,z:"t1",type:"test",wires:[]},
|
|
|
|
{id:"t1",type:"tab"}
|
|
|
|
];
|
2018-04-23 14:24:51 +01:00
|
|
|
flows.init({log:mockLog, settings:{},storage:storage});
|
2015-10-11 20:37:11 +01:00
|
|
|
flows.setFlows(originalConfig).then(function() {
|
2016-10-10 13:27:43 +01:00
|
|
|
credentialsClean.called.should.be.true();
|
|
|
|
storage.hasOwnProperty('conf').should.be.true();
|
2016-10-09 22:02:24 +01:00
|
|
|
flows.getFlows().flows.should.eql(originalConfig);
|
2015-10-11 20:37:11 +01:00
|
|
|
done();
|
|
|
|
});
|
|
|
|
|
|
|
|
});
|
2016-10-09 22:02:24 +01:00
|
|
|
it('loads the full flow for type load', function(done) {
|
2015-10-11 20:37:11 +01:00
|
|
|
var originalConfig = [
|
|
|
|
{id:"t1-1",x:10,y:10,z:"t1",type:"test",wires:[]},
|
|
|
|
{id:"t1",type:"tab"}
|
|
|
|
];
|
2016-10-09 22:02:24 +01:00
|
|
|
var loadStorage = {
|
|
|
|
saveFlows: function(conf) {
|
|
|
|
loadStorage.conf = conf;
|
2020-11-30 14:38:48 +00:00
|
|
|
return Promise.resolve(456);
|
2016-10-09 22:02:24 +01:00
|
|
|
},
|
|
|
|
getFlows: function() {
|
2020-11-30 14:38:48 +00:00
|
|
|
return Promise.resolve({flows:originalConfig,rev:123})
|
2016-10-09 22:02:24 +01:00
|
|
|
}
|
|
|
|
}
|
2018-04-23 14:24:51 +01:00
|
|
|
flows.init({log:mockLog, settings:{},storage:loadStorage});
|
2015-10-11 20:37:11 +01:00
|
|
|
flows.setFlows(originalConfig,"load").then(function() {
|
2016-10-10 13:27:43 +01:00
|
|
|
credentialsClean.called.should.be.false();
|
2015-10-11 20:37:11 +01:00
|
|
|
// 'load' type does not trigger a save
|
2016-10-10 13:27:43 +01:00
|
|
|
loadStorage.hasOwnProperty('conf').should.be.false();
|
2016-10-09 22:02:24 +01:00
|
|
|
flows.getFlows().flows.should.eql(originalConfig);
|
2015-10-11 20:37:11 +01:00
|
|
|
done();
|
|
|
|
});
|
|
|
|
|
|
|
|
});
|
|
|
|
|
|
|
|
it('extracts credentials from the full flow', function(done) {
|
|
|
|
var originalConfig = [
|
2016-09-21 21:58:50 +01:00
|
|
|
{id:"t1-1",x:10,y:10,z:"t1",type:"test",wires:[],credentials:{"a":1}},
|
2015-10-11 20:37:11 +01:00
|
|
|
{id:"t1",type:"tab"}
|
|
|
|
];
|
2018-04-23 14:24:51 +01:00
|
|
|
flows.init({log:mockLog, settings:{},storage:storage});
|
2015-10-11 20:37:11 +01:00
|
|
|
flows.setFlows(originalConfig).then(function() {
|
2016-10-10 13:27:43 +01:00
|
|
|
credentialsClean.called.should.be.true();
|
|
|
|
storage.hasOwnProperty('conf').should.be.true();
|
2015-10-11 20:37:11 +01:00
|
|
|
var cleanedFlows = flows.getFlows();
|
2016-10-09 22:02:24 +01:00
|
|
|
storage.conf.flows.should.eql(cleanedFlows.flows);
|
|
|
|
cleanedFlows.flows.should.not.eql(originalConfig);
|
|
|
|
cleanedFlows.flows[0].credentials = {"a":1};
|
|
|
|
cleanedFlows.flows.should.eql(originalConfig);
|
2015-10-11 20:37:11 +01:00
|
|
|
done();
|
|
|
|
});
|
2015-11-02 15:38:16 +00:00
|
|
|
});
|
2015-10-11 20:37:11 +01:00
|
|
|
|
2020-02-13 16:44:48 +00:00
|
|
|
it('sets the full flow including credentials', function(done) {
|
|
|
|
var originalConfig = [
|
|
|
|
{id:"t1-1",x:10,y:10,z:"t1",type:"test",wires:[]},
|
|
|
|
{id:"t1",type:"tab"}
|
|
|
|
];
|
|
|
|
var credentials = {"t1-1":{"a":1}};
|
|
|
|
|
|
|
|
flows.init({log:mockLog, settings:{},storage:storage});
|
|
|
|
flows.setFlows(originalConfig,credentials).then(function() {
|
2021-03-06 20:27:51 +00:00
|
|
|
credentialsClean.called.should.be.true();
|
|
|
|
credentialsAdd.called.should.be.true();
|
|
|
|
credentialsAdd.lastCall.args[0].should.eql("t1-1");
|
|
|
|
credentialsAdd.lastCall.args[1].should.eql({"a":1});
|
2020-02-13 16:44:48 +00:00
|
|
|
flows.getFlows().flows.should.eql(originalConfig);
|
|
|
|
done();
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
2015-11-02 15:38:16 +00:00
|
|
|
it('updates existing flows with partial deployment - nodes', function(done) {
|
|
|
|
var originalConfig = [
|
|
|
|
{id:"t1-1",x:10,y:10,z:"t1",type:"test",wires:[]},
|
|
|
|
{id:"t1",type:"tab"}
|
|
|
|
];
|
|
|
|
var newConfig = clone(originalConfig);
|
|
|
|
newConfig.push({id:"t1-2",x:10,y:10,z:"t1",type:"test",wires:[]});
|
|
|
|
newConfig.push({id:"t2",type:"tab"});
|
|
|
|
newConfig.push({id:"t2-1",x:10,y:10,z:"t2",type:"test",wires:[]});
|
|
|
|
storage.getFlows = function() {
|
2020-11-30 14:38:48 +00:00
|
|
|
return Promise.resolve({flows:originalConfig});
|
2015-11-02 15:38:16 +00:00
|
|
|
}
|
2020-09-29 16:29:10 +01:00
|
|
|
events.once('flows:started',function() {
|
2015-11-02 15:38:16 +00:00
|
|
|
flows.setFlows(newConfig,"nodes").then(function() {
|
2016-10-09 22:02:24 +01:00
|
|
|
flows.getFlows().flows.should.eql(newConfig);
|
2016-10-10 13:27:43 +01:00
|
|
|
flowCreate.flows['t1'].update.called.should.be.true();
|
|
|
|
flowCreate.flows['t2'].start.called.should.be.true();
|
|
|
|
flowCreate.flows['_GLOBAL_'].update.called.should.be.true();
|
2015-11-02 15:38:16 +00:00
|
|
|
done();
|
|
|
|
})
|
|
|
|
});
|
|
|
|
|
2018-04-23 14:24:51 +01:00
|
|
|
flows.init({log:mockLog, settings:{},storage:storage});
|
2015-11-02 15:38:16 +00:00
|
|
|
flows.load().then(function() {
|
|
|
|
flows.startFlows();
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
it('updates existing flows with partial deployment - flows', function(done) {
|
|
|
|
var originalConfig = [
|
|
|
|
{id:"t1-1",x:10,y:10,z:"t1",type:"test",wires:[]},
|
|
|
|
{id:"t1",type:"tab"}
|
|
|
|
];
|
|
|
|
var newConfig = clone(originalConfig);
|
|
|
|
newConfig.push({id:"t1-2",x:10,y:10,z:"t1",type:"test",wires:[]});
|
|
|
|
newConfig.push({id:"t2",type:"tab"});
|
|
|
|
newConfig.push({id:"t2-1",x:10,y:10,z:"t2",type:"test",wires:[]});
|
|
|
|
storage.getFlows = function() {
|
2020-11-30 14:38:48 +00:00
|
|
|
return Promise.resolve({flows:originalConfig});
|
2015-11-02 15:38:16 +00:00
|
|
|
}
|
|
|
|
|
2020-09-29 16:29:10 +01:00
|
|
|
events.once('flows:started',function() {
|
2015-11-02 15:38:16 +00:00
|
|
|
flows.setFlows(newConfig,"nodes").then(function() {
|
2016-10-09 22:02:24 +01:00
|
|
|
flows.getFlows().flows.should.eql(newConfig);
|
2016-10-10 13:27:43 +01:00
|
|
|
flowCreate.flows['t1'].update.called.should.be.true();
|
|
|
|
flowCreate.flows['t2'].start.called.should.be.true();
|
|
|
|
flowCreate.flows['_GLOBAL_'].update.called.should.be.true();
|
2015-11-02 15:38:16 +00:00
|
|
|
flows.stopFlows().then(done);
|
|
|
|
})
|
|
|
|
});
|
|
|
|
|
2018-04-23 14:24:51 +01:00
|
|
|
flows.init({log:mockLog, settings:{},storage:storage});
|
2015-11-02 15:38:16 +00:00
|
|
|
flows.load().then(function() {
|
|
|
|
flows.startFlows();
|
|
|
|
});
|
2015-10-11 20:37:11 +01:00
|
|
|
});
|
|
|
|
|
2020-02-13 16:44:48 +00:00
|
|
|
it('returns error if it cannot decrypt credentials', function(done) {
|
|
|
|
var originalConfig = [
|
|
|
|
{id:"t1-1",x:10,y:10,z:"t1",type:"test",wires:[]},
|
|
|
|
{id:"t1",type:"tab"}
|
|
|
|
];
|
|
|
|
var credentials = {"$":"fail"};
|
|
|
|
|
|
|
|
flows.init({log:mockLog, settings:{},storage:storage});
|
|
|
|
flows.setFlows(originalConfig,credentials).then(function() {
|
|
|
|
done("Unexpected success when credentials couldn't be decrypted")
|
|
|
|
}).catch(function(err) {
|
|
|
|
done();
|
|
|
|
});
|
|
|
|
});
|
2015-10-11 20:37:11 +01:00
|
|
|
});
|
|
|
|
|
|
|
|
describe('#load', function() {
|
|
|
|
it('loads the flow config', function(done) {
|
|
|
|
var originalConfig = [
|
|
|
|
{id:"t1-1",x:10,y:10,z:"t1",type:"test",wires:[]},
|
|
|
|
{id:"t1",type:"tab"}
|
|
|
|
];
|
|
|
|
storage.getFlows = function() {
|
2020-11-30 14:38:48 +00:00
|
|
|
return Promise.resolve({flows:originalConfig});
|
2015-10-11 20:37:11 +01:00
|
|
|
}
|
2018-04-23 14:24:51 +01:00
|
|
|
flows.init({log:mockLog, settings:{},storage:storage});
|
2015-10-11 20:37:11 +01:00
|
|
|
flows.load().then(function() {
|
2016-10-10 13:27:43 +01:00
|
|
|
credentialsLoad.called.should.be.true();
|
2015-10-11 20:37:11 +01:00
|
|
|
// 'load' type does not trigger a save
|
2016-10-10 13:27:43 +01:00
|
|
|
storage.hasOwnProperty('conf').should.be.false();
|
2016-10-09 22:02:24 +01:00
|
|
|
flows.getFlows().flows.should.eql(originalConfig);
|
2015-10-11 20:37:11 +01:00
|
|
|
done();
|
|
|
|
});
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
describe('#startFlows', function() {
|
2015-11-02 15:38:16 +00:00
|
|
|
it('starts the loaded config', function(done) {
|
|
|
|
var originalConfig = [
|
|
|
|
{id:"t1-1",x:10,y:10,z:"t1",type:"test",wires:[]},
|
|
|
|
{id:"t1",type:"tab"}
|
|
|
|
];
|
|
|
|
storage.getFlows = function() {
|
2020-11-30 14:38:48 +00:00
|
|
|
return Promise.resolve({flows:originalConfig});
|
2015-11-02 15:38:16 +00:00
|
|
|
}
|
|
|
|
|
2020-09-29 16:29:10 +01:00
|
|
|
events.once('flows:started',function() {
|
2015-11-02 15:38:16 +00:00
|
|
|
Object.keys(flowCreate.flows).should.eql(['_GLOBAL_','t1']);
|
|
|
|
done();
|
|
|
|
});
|
|
|
|
|
2018-04-23 14:24:51 +01:00
|
|
|
flows.init({log:mockLog, settings:{},storage:storage});
|
2015-11-02 15:38:16 +00:00
|
|
|
flows.load().then(function() {
|
2021-02-13 00:18:04 +00:00
|
|
|
return flows.startFlows();
|
2015-11-02 15:38:16 +00:00
|
|
|
});
|
|
|
|
});
|
|
|
|
it('does not start if nodes missing', function(done) {
|
2015-10-11 20:37:11 +01:00
|
|
|
var originalConfig = [
|
|
|
|
{id:"t1-1",x:10,y:10,z:"t1",type:"test",wires:[]},
|
2015-11-02 15:38:16 +00:00
|
|
|
{id:"t1-2",x:10,y:10,z:"t1",type:"missing",wires:[]},
|
2015-10-11 20:37:11 +01:00
|
|
|
{id:"t1",type:"tab"}
|
|
|
|
];
|
|
|
|
storage.getFlows = function() {
|
2020-11-30 14:38:48 +00:00
|
|
|
return Promise.resolve({flows:originalConfig});
|
2015-10-11 20:37:11 +01:00
|
|
|
}
|
2015-11-02 15:38:16 +00:00
|
|
|
|
2018-04-23 14:24:51 +01:00
|
|
|
flows.init({log:mockLog, settings:{},storage:storage});
|
2015-10-11 20:37:11 +01:00
|
|
|
flows.load().then(function() {
|
2021-02-13 00:18:04 +00:00
|
|
|
return flows.startFlows();
|
|
|
|
}).then(() => {
|
|
|
|
try {
|
|
|
|
flowCreate.called.should.be.false();
|
|
|
|
done();
|
|
|
|
} catch(err) {
|
|
|
|
done(err);
|
|
|
|
}
|
2015-10-11 20:37:11 +01:00
|
|
|
});
|
|
|
|
});
|
2015-11-02 15:38:16 +00:00
|
|
|
|
|
|
|
it('starts when missing nodes registered', function(done) {
|
|
|
|
var originalConfig = [
|
|
|
|
{id:"t1-1",x:10,y:10,z:"t1",type:"test",wires:[]},
|
|
|
|
{id:"t1-2",x:10,y:10,z:"t1",type:"missing",wires:[]},
|
|
|
|
{id:"t1-3",x:10,y:10,z:"t1",type:"missing2",wires:[]},
|
|
|
|
{id:"t1",type:"tab"}
|
|
|
|
];
|
|
|
|
storage.getFlows = function() {
|
2020-11-30 14:38:48 +00:00
|
|
|
return Promise.resolve({flows:originalConfig});
|
2015-11-02 15:38:16 +00:00
|
|
|
}
|
2018-04-23 14:24:51 +01:00
|
|
|
flows.init({log:mockLog, settings:{},storage:storage});
|
2015-11-02 15:38:16 +00:00
|
|
|
flows.load().then(function() {
|
2021-02-13 00:18:04 +00:00
|
|
|
return flows.startFlows();
|
|
|
|
}).then(() => {
|
2016-10-10 13:27:43 +01:00
|
|
|
flowCreate.called.should.be.false();
|
2015-11-02 15:38:16 +00:00
|
|
|
events.emit("type-registered","missing");
|
|
|
|
setTimeout(function() {
|
2016-10-10 13:27:43 +01:00
|
|
|
flowCreate.called.should.be.false();
|
2015-11-02 15:38:16 +00:00
|
|
|
events.emit("type-registered","missing2");
|
|
|
|
setTimeout(function() {
|
2016-10-10 13:27:43 +01:00
|
|
|
flowCreate.called.should.be.true();
|
2015-11-02 15:38:16 +00:00
|
|
|
done();
|
|
|
|
},10);
|
|
|
|
},10);
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
2021-02-13 00:18:04 +00:00
|
|
|
it('does not start if external modules missing', function(done) {
|
|
|
|
var originalConfig = [
|
|
|
|
{id:"node-with-missing-modules",x:10,y:10,z:"t1",type:"test",wires:[]},
|
|
|
|
{id:"t1",type:"tab"}
|
|
|
|
];
|
|
|
|
|
|
|
|
storage.getFlows = function() {
|
|
|
|
return Promise.resolve({flows:originalConfig});
|
|
|
|
}
|
|
|
|
var receivedEvent = null;
|
|
|
|
var handleEvent = function(payload) {
|
|
|
|
receivedEvent = payload;
|
|
|
|
}
|
|
|
|
|
|
|
|
events.on("runtime-event",handleEvent);
|
|
|
|
|
|
|
|
//{id:"runtime-state",payload:{error:"missing-modules", type:"warning",text:"notification.warnings.missing-modules",modules:missingModules},retain:true});"
|
|
|
|
|
2015-11-02 15:38:16 +00:00
|
|
|
|
2021-02-13 00:18:04 +00:00
|
|
|
flows.init({log:mockLog, settings:{},storage:storage});
|
|
|
|
flows.load().then(flows.startFlows).then(() => {
|
|
|
|
events.removeListener("runtime-event",handleEvent);
|
|
|
|
try {
|
|
|
|
flowCreate.called.should.be.false();
|
|
|
|
receivedEvent.should.have.property('id','runtime-state');
|
|
|
|
receivedEvent.should.have.property('payload',
|
|
|
|
{ error: 'missing-modules',
|
|
|
|
type: 'warning',
|
|
|
|
text: 'notification.warnings.missing-modules',
|
|
|
|
modules: [] }
|
|
|
|
);
|
|
|
|
|
|
|
|
done();
|
|
|
|
}catch(err) {
|
|
|
|
done(err)
|
|
|
|
}
|
|
|
|
});
|
|
|
|
});
|
2015-11-02 15:38:16 +00:00
|
|
|
|
2015-10-11 20:37:11 +01:00
|
|
|
});
|
|
|
|
|
2016-09-21 21:58:50 +01:00
|
|
|
describe.skip('#get',function() {
|
2015-10-11 20:37:11 +01:00
|
|
|
|
|
|
|
});
|
|
|
|
|
|
|
|
describe('#eachNode', function() {
|
2015-11-02 15:38:16 +00:00
|
|
|
it('iterates the flow nodes', function(done) {
|
|
|
|
var originalConfig = [
|
|
|
|
{id:"t1-1",x:10,y:10,z:"t1",type:"test",wires:[]},
|
|
|
|
{id:"t1",type:"tab"}
|
|
|
|
];
|
|
|
|
storage.getFlows = function() {
|
2020-11-30 14:38:48 +00:00
|
|
|
return Promise.resolve({flows:originalConfig});
|
2015-11-02 15:38:16 +00:00
|
|
|
}
|
2018-04-23 14:24:51 +01:00
|
|
|
flows.init({log:mockLog, settings:{},storage:storage});
|
2015-11-02 15:38:16 +00:00
|
|
|
flows.load().then(function() {
|
|
|
|
var c = 0;
|
|
|
|
flows.eachNode(function(node) {
|
|
|
|
c++
|
|
|
|
})
|
|
|
|
c.should.equal(2);
|
|
|
|
done();
|
|
|
|
});
|
|
|
|
});
|
2015-10-11 20:37:11 +01:00
|
|
|
});
|
|
|
|
|
|
|
|
describe('#stopFlows', function() {
|
|
|
|
|
|
|
|
});
|
2019-01-16 16:27:19 +00:00
|
|
|
// describe('#handleError', function() {
|
|
|
|
// it('passes error to correct flow', function(done) {
|
|
|
|
// var originalConfig = [
|
|
|
|
// {id:"t1-1",x:10,y:10,z:"t1",type:"test",wires:[]},
|
|
|
|
// {id:"t1",type:"tab"}
|
|
|
|
// ];
|
|
|
|
// storage.getFlows = function() {
|
2020-11-30 14:38:48 +00:00
|
|
|
// return Promise.resolve({flows:originalConfig});
|
2019-01-16 16:27:19 +00:00
|
|
|
// }
|
|
|
|
//
|
2020-09-29 16:29:10 +01:00
|
|
|
// events.once('flows:started',function() {
|
2019-01-16 16:27:19 +00:00
|
|
|
// flows.handleError(originalConfig[0],"message",{});
|
|
|
|
// flowCreate.flows['t1'].handleError.called.should.be.true();
|
|
|
|
// done();
|
|
|
|
// });
|
|
|
|
//
|
|
|
|
// flows.init({log:mockLog, settings:{},storage:storage});
|
|
|
|
// flows.load().then(function() {
|
|
|
|
// flows.startFlows();
|
|
|
|
// });
|
|
|
|
// });
|
|
|
|
// it('passes error to flows that use the originating global config', function(done) {
|
|
|
|
// var originalConfig = [
|
|
|
|
// {id:"configNode",type:"test"},
|
|
|
|
// {id:"t1",type:"tab"},
|
|
|
|
// {id:"t1-1",x:10,y:10,z:"t1",type:"test",config:"configNode",wires:[]},
|
|
|
|
// {id:"t2",type:"tab"},
|
|
|
|
// {id:"t2-1",x:10,y:10,z:"t2",type:"test",wires:[]},
|
|
|
|
// {id:"t3",type:"tab"},
|
|
|
|
// {id:"t3-1",x:10,y:10,z:"t3",type:"test",config:"configNode",wires:[]}
|
|
|
|
// ];
|
|
|
|
// storage.getFlows = function() {
|
2020-11-30 14:38:48 +00:00
|
|
|
// return Promise.resolve({flows:originalConfig});
|
2019-01-16 16:27:19 +00:00
|
|
|
// }
|
|
|
|
//
|
2020-09-29 16:29:10 +01:00
|
|
|
// events.once('flows:started',function() {
|
2019-01-16 16:27:19 +00:00
|
|
|
// flows.handleError(originalConfig[0],"message",{});
|
|
|
|
// try {
|
|
|
|
// flowCreate.flows['t1'].handleError.called.should.be.true();
|
|
|
|
// flowCreate.flows['t2'].handleError.called.should.be.false();
|
|
|
|
// flowCreate.flows['t3'].handleError.called.should.be.true();
|
|
|
|
// done();
|
|
|
|
// } catch(err) {
|
|
|
|
// done(err);
|
|
|
|
// }
|
|
|
|
// });
|
|
|
|
//
|
|
|
|
// flows.init({log:mockLog, settings:{},storage:storage});
|
|
|
|
// flows.load().then(function() {
|
|
|
|
// flows.startFlows();
|
|
|
|
// });
|
|
|
|
// });
|
|
|
|
// });
|
|
|
|
// describe('#handleStatus', function() {
|
|
|
|
// it('passes status to correct flow', function(done) {
|
|
|
|
// var originalConfig = [
|
|
|
|
// {id:"t1-1",x:10,y:10,z:"t1",type:"test",wires:[]},
|
|
|
|
// {id:"t1",type:"tab"}
|
|
|
|
// ];
|
|
|
|
// storage.getFlows = function() {
|
2020-11-30 14:38:48 +00:00
|
|
|
// return Promise.resolve({flows:originalConfig});
|
2019-01-16 16:27:19 +00:00
|
|
|
// }
|
|
|
|
//
|
2020-09-29 16:29:10 +01:00
|
|
|
// events.once('flows:started',function() {
|
2019-01-16 16:27:19 +00:00
|
|
|
// flows.handleStatus(originalConfig[0],"message");
|
|
|
|
// flowCreate.flows['t1'].handleStatus.called.should.be.true();
|
|
|
|
// done();
|
|
|
|
// });
|
|
|
|
//
|
|
|
|
// flows.init({log:mockLog, settings:{},storage:storage});
|
|
|
|
// flows.load().then(function() {
|
|
|
|
// flows.startFlows();
|
|
|
|
// });
|
|
|
|
// });
|
|
|
|
//
|
|
|
|
// it('passes status to flows that use the originating global config', function(done) {
|
|
|
|
// var originalConfig = [
|
|
|
|
// {id:"configNode",type:"test"},
|
|
|
|
// {id:"t1",type:"tab"},
|
|
|
|
// {id:"t1-1",x:10,y:10,z:"t1",type:"test",config:"configNode",wires:[]},
|
|
|
|
// {id:"t2",type:"tab"},
|
|
|
|
// {id:"t2-1",x:10,y:10,z:"t2",type:"test",wires:[]},
|
|
|
|
// {id:"t3",type:"tab"},
|
|
|
|
// {id:"t3-1",x:10,y:10,z:"t3",type:"test",config:"configNode",wires:[]}
|
|
|
|
// ];
|
|
|
|
// storage.getFlows = function() {
|
2020-11-30 14:38:48 +00:00
|
|
|
// return Promise.resolve({flows:originalConfig});
|
2019-01-16 16:27:19 +00:00
|
|
|
// }
|
|
|
|
//
|
2020-09-29 16:29:10 +01:00
|
|
|
// events.once('flows:started',function() {
|
2019-01-16 16:27:19 +00:00
|
|
|
// flows.handleStatus(originalConfig[0],"message");
|
|
|
|
// try {
|
|
|
|
// flowCreate.flows['t1'].handleStatus.called.should.be.true();
|
|
|
|
// flowCreate.flows['t2'].handleStatus.called.should.be.false();
|
|
|
|
// flowCreate.flows['t3'].handleStatus.called.should.be.true();
|
|
|
|
// done();
|
|
|
|
// } catch(err) {
|
|
|
|
// done(err);
|
|
|
|
// }
|
|
|
|
// });
|
|
|
|
//
|
|
|
|
// flows.init({log:mockLog, settings:{},storage:storage});
|
|
|
|
// flows.load().then(function() {
|
|
|
|
// flows.startFlows();
|
|
|
|
// });
|
|
|
|
// });
|
|
|
|
// });
|
2015-11-09 11:29:48 +00:00
|
|
|
|
|
|
|
describe('#checkTypeInUse', function() {
|
|
|
|
|
|
|
|
before(function() {
|
2021-04-09 11:22:57 +01:00
|
|
|
sinon.stub(typeRegistry,"getNodeInfo").callsFake(function(id) {
|
2015-11-09 11:29:48 +00:00
|
|
|
if (id === 'unused-module') {
|
|
|
|
return {types:['one','two','three']}
|
|
|
|
} else {
|
|
|
|
return {types:['one','test','three']}
|
|
|
|
}
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
after(function() {
|
|
|
|
typeRegistry.getNodeInfo.restore();
|
|
|
|
});
|
|
|
|
|
|
|
|
it('returns cleanly if type not is use', function(done) {
|
|
|
|
var originalConfig = [
|
|
|
|
{id:"t1-1",x:10,y:10,z:"t1",type:"test",wires:[]},
|
|
|
|
{id:"t1",type:"tab"}
|
|
|
|
];
|
2018-04-23 14:24:51 +01:00
|
|
|
flows.init({log:mockLog, settings:{},storage:storage});
|
2015-11-09 11:29:48 +00:00
|
|
|
flows.setFlows(originalConfig).then(function() {
|
|
|
|
flows.checkTypeInUse("unused-module");
|
|
|
|
done();
|
|
|
|
});
|
|
|
|
});
|
|
|
|
it('throws error if type is in use', function(done) {
|
|
|
|
var originalConfig = [
|
|
|
|
{id:"t1-1",x:10,y:10,z:"t1",type:"test",wires:[]},
|
|
|
|
{id:"t1",type:"tab"}
|
|
|
|
];
|
2018-04-23 14:24:51 +01:00
|
|
|
flows.init({log:mockLog, settings:{},storage:storage});
|
2015-11-09 11:29:48 +00:00
|
|
|
flows.setFlows(originalConfig).then(function() {
|
|
|
|
/*jshint immed: false */
|
|
|
|
try {
|
|
|
|
flows.checkTypeInUse("used-module");
|
|
|
|
done("type_in_use error not thrown");
|
|
|
|
} catch(err) {
|
|
|
|
err.code.should.eql("type_in_use");
|
|
|
|
done();
|
|
|
|
}
|
|
|
|
});
|
|
|
|
});
|
|
|
|
});
|
2015-12-06 22:49:51 +00:00
|
|
|
|
|
|
|
describe('#addFlow', function() {
|
|
|
|
it("rejects duplicate node id",function(done) {
|
|
|
|
var originalConfig = [
|
|
|
|
{id:"t1-1",x:10,y:10,z:"t1",type:"test",wires:[]},
|
|
|
|
{id:"t1",type:"tab"}
|
|
|
|
];
|
|
|
|
storage.getFlows = function() {
|
2020-11-30 14:38:48 +00:00
|
|
|
return Promise.resolve({flows:originalConfig});
|
2015-12-06 22:49:51 +00:00
|
|
|
}
|
2018-04-23 14:24:51 +01:00
|
|
|
flows.init({log:mockLog, settings:{},storage:storage});
|
2015-12-06 22:49:51 +00:00
|
|
|
flows.load().then(function() {
|
|
|
|
flows.addFlow({
|
|
|
|
label:'new flow',
|
|
|
|
nodes:[
|
|
|
|
{id:"t1-1",x:10,y:10,z:"t1",type:"test",wires:[]}
|
|
|
|
]
|
|
|
|
}).then(function() {
|
|
|
|
done(new Error('failed to reject duplicate node id'));
|
2018-04-24 15:01:49 +01:00
|
|
|
}).catch(function(err) {
|
2015-12-06 22:49:51 +00:00
|
|
|
done();
|
|
|
|
})
|
|
|
|
});
|
|
|
|
|
|
|
|
});
|
|
|
|
|
|
|
|
it("addFlow",function(done) {
|
|
|
|
var originalConfig = [
|
|
|
|
{id:"t1-1",x:10,y:10,z:"t1",type:"test",wires:[]},
|
|
|
|
{id:"t1",type:"tab"}
|
|
|
|
];
|
|
|
|
storage.getFlows = function() {
|
2020-11-30 14:38:48 +00:00
|
|
|
return Promise.resolve({flows:originalConfig});
|
2015-12-06 22:49:51 +00:00
|
|
|
}
|
2015-12-09 21:51:46 +00:00
|
|
|
storage.setFlows = function() {
|
2020-11-30 14:38:48 +00:00
|
|
|
return Promise.resolve();
|
2015-12-09 21:51:46 +00:00
|
|
|
}
|
2018-04-23 14:24:51 +01:00
|
|
|
flows.init({log:mockLog, settings:{},storage:storage});
|
2015-12-06 22:49:51 +00:00
|
|
|
flows.load().then(function() {
|
|
|
|
return flows.startFlows();
|
|
|
|
}).then(function() {
|
|
|
|
flows.addFlow({
|
|
|
|
label:'new flow',
|
|
|
|
nodes:[
|
|
|
|
{id:"t2-1",x:10,y:10,z:"t1",type:"test",wires:[]},
|
2018-04-23 14:24:51 +01:00
|
|
|
{id:"t2-2",x:10,y:10,z:"t1",type:"test",wires:[]},
|
2015-12-06 22:49:51 +00:00
|
|
|
{id:"t2-3",z:"t1",type:"test"}
|
|
|
|
]
|
|
|
|
}).then(function(id) {
|
2016-10-09 22:02:24 +01:00
|
|
|
flows.getFlows().flows.should.have.lengthOf(6);
|
2015-12-06 22:49:51 +00:00
|
|
|
var createdFlows = Object.keys(flowCreate.flows);
|
|
|
|
createdFlows.should.have.lengthOf(3);
|
|
|
|
createdFlows[2].should.eql(id);
|
|
|
|
done();
|
2018-04-24 15:01:49 +01:00
|
|
|
}).catch(function(err) {
|
2015-12-06 22:49:51 +00:00
|
|
|
done(err);
|
|
|
|
})
|
|
|
|
});
|
|
|
|
|
|
|
|
});
|
|
|
|
})
|
|
|
|
describe('#updateFlow', function() {
|
|
|
|
it.skip("updateFlow");
|
|
|
|
})
|
|
|
|
describe('#removeFlow', function() {
|
|
|
|
it.skip("removeFlow");
|
|
|
|
})
|
|
|
|
describe('#disableFlow', function() {
|
|
|
|
it.skip("disableFlow");
|
|
|
|
})
|
|
|
|
describe('#enableFlow', function() {
|
|
|
|
it.skip("enableFlow");
|
|
|
|
})
|
2015-11-02 20:41:59 +00:00
|
|
|
});
|