2014-07-14 21:44:34 +01:00
|
|
|
/**
|
|
|
|
* Copyright 2014 IBM Corp.
|
|
|
|
*
|
|
|
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
* you may not use this file except in compliance with the License.
|
|
|
|
* You may obtain a copy of the License at
|
|
|
|
*
|
|
|
|
* http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
*
|
|
|
|
* Unless required by applicable law or agreed to in writing, software
|
|
|
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
* See the License for the specific language governing permissions and
|
|
|
|
* limitations under the License.
|
|
|
|
**/
|
|
|
|
|
|
|
|
var should = require("should");
|
2014-08-28 00:35:07 +01:00
|
|
|
var sinon = require("sinon");
|
2014-07-14 21:44:34 +01:00
|
|
|
var when = require("when");
|
2015-03-22 20:12:10 +00:00
|
|
|
var clone = require("clone");
|
2014-07-17 08:06:30 +01:00
|
|
|
var flows = require("../../../red/nodes/flows");
|
|
|
|
var RedNode = require("../../../red/nodes/Node");
|
|
|
|
var RED = require("../../../red/nodes");
|
|
|
|
var events = require("../../../red/events");
|
2015-03-22 20:12:10 +00:00
|
|
|
var credentials = require("../../../red/nodes/credentials");
|
2014-08-28 00:35:07 +01:00
|
|
|
var typeRegistry = require("../../../red/nodes/registry");
|
2015-03-22 20:12:10 +00:00
|
|
|
var Flow = require("../../../red/nodes/Flow");
|
2014-08-28 00:35:07 +01:00
|
|
|
|
|
|
|
|
|
|
|
var settings = {
|
|
|
|
available: function() { return false; }
|
|
|
|
}
|
2014-07-14 21:44:34 +01:00
|
|
|
|
|
|
|
function loadFlows(testFlows, cb) {
|
|
|
|
var storage = {
|
|
|
|
getFlows: function() {
|
2014-08-28 00:35:07 +01:00
|
|
|
return when.resolve(testFlows);
|
2014-07-14 21:44:34 +01:00
|
|
|
},
|
|
|
|
getCredentials: function() {
|
2014-08-28 00:35:07 +01:00
|
|
|
return when.resolve({});
|
|
|
|
}
|
2014-07-14 21:44:34 +01:00
|
|
|
};
|
2014-08-28 00:35:07 +01:00
|
|
|
RED.init(settings, storage);
|
2014-07-14 21:44:34 +01:00
|
|
|
flows.load().then(function() {
|
|
|
|
should.deepEqual(testFlows, flows.getFlows());
|
|
|
|
cb();
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
describe('flows', function() {
|
|
|
|
|
2014-10-29 21:38:05 +00:00
|
|
|
afterEach(function(done) {
|
2015-01-10 22:09:37 +00:00
|
|
|
flows.stopFlows().then(function() {
|
2014-10-29 21:38:05 +00:00
|
|
|
loadFlows([],done);
|
|
|
|
});
|
|
|
|
});
|
2014-07-14 21:44:34 +01:00
|
|
|
|
|
|
|
describe('#load',function() {
|
2015-01-10 22:09:37 +00:00
|
|
|
|
2014-07-14 21:44:34 +01:00
|
|
|
it('should load nothing when storage is empty',function(done) {
|
|
|
|
loadFlows([], done);
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should load and start an empty tab flow',function(done) {
|
2014-08-28 00:35:07 +01:00
|
|
|
loadFlows([{"type":"tab","id":"tab1","label":"Sheet 1"}], function() {});
|
2014-07-14 21:44:34 +01:00
|
|
|
events.once('nodes-started', function() { done(); });
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should load and start a registered node type', function(done) {
|
|
|
|
RED.registerType('debug', function() {});
|
2014-08-28 00:35:07 +01:00
|
|
|
var typeRegistryGet = sinon.stub(typeRegistry,"get",function(nt) {
|
2015-01-10 22:09:37 +00:00
|
|
|
return RedNode;
|
2014-08-28 00:35:07 +01:00
|
|
|
});
|
2014-07-14 21:44:34 +01:00
|
|
|
loadFlows([{"id":"n1","type":"debug"}], function() { });
|
2014-08-28 00:35:07 +01:00
|
|
|
events.once('nodes-started', function() {
|
|
|
|
typeRegistryGet.restore();
|
|
|
|
done();
|
|
|
|
});
|
2014-07-14 21:44:34 +01:00
|
|
|
});
|
|
|
|
|
2014-08-28 00:35:07 +01:00
|
|
|
it('should load and start when node type is registered', function(done) {
|
|
|
|
var typeRegistryGet = sinon.stub(typeRegistry,"get");
|
|
|
|
typeRegistryGet.onCall(0).returns(null);
|
2015-01-10 22:09:37 +00:00
|
|
|
typeRegistryGet.returns(RedNode);
|
2014-08-28 00:35:07 +01:00
|
|
|
loadFlows([{"id":"n2","type":"inject"}], function() {
|
|
|
|
events.emit('type-registered','inject');
|
|
|
|
});
|
|
|
|
events.once('nodes-started', function() {
|
|
|
|
typeRegistryGet.restore();
|
|
|
|
done();
|
|
|
|
});
|
2014-07-14 21:44:34 +01:00
|
|
|
});
|
2014-10-29 21:38:05 +00:00
|
|
|
|
|
|
|
it('should not instantiate nodes of an unused subflow', function(done) {
|
|
|
|
RED.registerType('abc', function() {});
|
|
|
|
var typeRegistryGet = sinon.stub(typeRegistry,"get",function(nt) {
|
2015-01-10 22:09:37 +00:00
|
|
|
return RedNode;
|
2014-10-29 21:38:05 +00:00
|
|
|
});
|
|
|
|
loadFlows([{"id":"n1","type":"subflow",inputs:[],outputs:[],wires:[]},
|
|
|
|
{"id":"n2","type":"abc","z":"n1",wires:[]}
|
|
|
|
],function() { });
|
|
|
|
events.once('nodes-started', function() {
|
|
|
|
(flows.get("n2") == null).should.be.true;
|
|
|
|
var ncount = 0
|
2015-01-10 22:09:37 +00:00
|
|
|
flows.eachNode(function(n) {
|
2014-10-29 21:38:05 +00:00
|
|
|
ncount++;
|
|
|
|
});
|
|
|
|
ncount.should.equal(0);
|
|
|
|
typeRegistryGet.restore();
|
|
|
|
done();
|
|
|
|
});
|
|
|
|
});
|
|
|
|
it('should instantiate nodes of an used subflow with new IDs', function(done) {
|
|
|
|
RED.registerType('abc', function() {});
|
|
|
|
var typeRegistryGet = sinon.stub(typeRegistry,"get",function(nt) {
|
|
|
|
return RedNode;
|
|
|
|
});
|
|
|
|
loadFlows([{"id":"n1","type":"subflow",inputs:[],outputs:[]},
|
|
|
|
{"id":"n2","type":"abc","z":"n1","name":"def",wires:[]},
|
|
|
|
{"id":"n3","type":"subflow:n1"}
|
|
|
|
], function() { });
|
|
|
|
events.once('nodes-started', function() {
|
|
|
|
// n2 should not get instantiated with that id
|
|
|
|
(flows.get("n2") == null).should.be.true;
|
|
|
|
var ncount = 0
|
|
|
|
var nodes = [];
|
2015-01-10 22:09:37 +00:00
|
|
|
flows.eachNode(function(n) {
|
2014-10-29 21:38:05 +00:00
|
|
|
nodes.push(n);
|
|
|
|
});
|
|
|
|
nodes.should.have.lengthOf(2);
|
|
|
|
|
|
|
|
// Assume the nodes are instantiated in this order - not
|
|
|
|
// a requirement, but makes the test easier to write.
|
|
|
|
nodes[0].should.have.property("id","n3");
|
|
|
|
nodes[0].should.have.property("type","subflow:n1");
|
|
|
|
nodes[1].should.not.have.property("id","n2");
|
|
|
|
nodes[1].should.have.property("name","def");
|
|
|
|
|
|
|
|
// TODO: verify instance wiring is correct
|
|
|
|
typeRegistryGet.restore();
|
|
|
|
done();
|
|
|
|
});
|
|
|
|
});
|
2014-07-14 21:44:34 +01:00
|
|
|
});
|
|
|
|
|
2014-07-14 22:00:09 +01:00
|
|
|
describe('#setFlows',function() {
|
2015-03-22 20:12:10 +00:00
|
|
|
var credentialsExtact;
|
|
|
|
var credentialsSave;
|
|
|
|
var stopFlows;
|
|
|
|
var startFlows;
|
|
|
|
var credentialsExtractNode;
|
|
|
|
beforeEach(function() {
|
|
|
|
credentialsExtact = sinon.stub(credentials,"extract",function(node) {credentialsExtractNode = clone(node);delete node.credentials;});
|
|
|
|
credentialsSave = sinon.stub(credentials,"save",function() { return when.resolve();});
|
|
|
|
stopFlows = sinon.stub(flows,"stopFlows",function() {return when.resolve();});
|
|
|
|
startFlows = sinon.stub(flows,"startFlows",function() {});
|
|
|
|
});
|
|
|
|
afterEach(function() {
|
|
|
|
credentialsExtact.restore();
|
|
|
|
credentialsSave.restore();
|
|
|
|
startFlows.restore();
|
|
|
|
stopFlows.restore();
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should extract credentials from nodes', function(done) {
|
|
|
|
var testFlow = [{"type":"testNode","credentials":{"a":1}},{"type":"testNode2"}];
|
|
|
|
var resultFlow = clone(testFlow);
|
|
|
|
var storage = { saveFlows: sinon.spy() };
|
2015-06-17 14:18:47 +01:00
|
|
|
flows.init({},storage);
|
2015-03-22 20:12:10 +00:00
|
|
|
flows.setFlows(testFlow,"full").then(function() {
|
|
|
|
try {
|
|
|
|
credentialsExtact.calledOnce.should.be.true;
|
|
|
|
// credential property stripped
|
|
|
|
testFlow.should.not.have.property("credentials");
|
|
|
|
credentialsExtractNode.should.eql(resultFlow[0]);
|
|
|
|
credentialsExtractNode.should.not.equal(resultFlow[0]);
|
|
|
|
|
|
|
|
credentialsSave.calledOnce.should.be.true;
|
|
|
|
|
|
|
|
storage.saveFlows.calledOnce.should.be.true;
|
|
|
|
storage.saveFlows.args[0][0].should.eql(testFlow);
|
|
|
|
|
|
|
|
stopFlows.calledOnce.should.be.true;
|
|
|
|
startFlows.calledOnce.should.be.true;
|
|
|
|
|
|
|
|
done();
|
|
|
|
} catch(err) {
|
|
|
|
done(err);
|
2014-07-14 22:00:09 +01:00
|
|
|
}
|
2015-03-22 20:12:10 +00:00
|
|
|
});
|
2014-07-14 22:00:09 +01:00
|
|
|
});
|
2015-03-22 20:12:10 +00:00
|
|
|
|
|
|
|
it('should apply diff on partial deployment', function(done) {
|
|
|
|
var testFlow = [{"type":"testNode"},{"type":"testNode2"}];
|
|
|
|
var testFlow2 = [{"type":"testNode3"},{"type":"testNode4"}];
|
|
|
|
var storage = { saveFlows: sinon.spy() };
|
2015-06-17 14:18:47 +01:00
|
|
|
flows.init({},storage);
|
2015-03-22 20:12:10 +00:00
|
|
|
|
|
|
|
flows.setFlows(testFlow,"full").then(function() {
|
|
|
|
flows.setFlows(testFlow2,"nodes").then(function() {
|
|
|
|
try {
|
|
|
|
credentialsExtact.called.should.be.false;
|
|
|
|
|
|
|
|
storage.saveFlows.calledTwice.should.be.true;
|
|
|
|
storage.saveFlows.args[1][0].should.eql(testFlow2);
|
|
|
|
|
|
|
|
stopFlows.calledTwice.should.be.true;
|
|
|
|
startFlows.calledTwice.should.be.true;
|
|
|
|
|
|
|
|
var configDiff = {
|
|
|
|
type: 'nodes',
|
|
|
|
stop: [],
|
|
|
|
rewire: [],
|
|
|
|
config: testFlow2
|
|
|
|
}
|
|
|
|
stopFlows.args[1][0].should.eql(configDiff);
|
|
|
|
startFlows.args[1][0].should.eql(configDiff);
|
|
|
|
|
|
|
|
done();
|
|
|
|
} catch(err) {
|
|
|
|
done(err);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
|
2014-07-14 22:00:09 +01:00
|
|
|
});
|
|
|
|
|
2014-07-14 21:44:34 +01:00
|
|
|
});
|