2015-11-09 12:29:48 +01:00
|
|
|
/**
|
2017-01-11 16:24:33 +01:00
|
|
|
* Copyright JS Foundation and other contributors, http://js.foundation
|
2015-11-09 12:29:48 +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 when = require("when");
|
2015-11-09 17:52:14 +01:00
|
|
|
var path = require("path");
|
2015-11-14 21:27:04 +01:00
|
|
|
var fs = require('fs');
|
2018-01-24 23:56:54 +01:00
|
|
|
var EventEmitter = require('events');
|
2015-11-09 12:29:48 +01:00
|
|
|
|
|
|
|
var child_process = require('child_process');
|
2018-04-26 13:32:05 +02:00
|
|
|
var installer = require("../../../red/runtime-registry/installer");
|
|
|
|
var registry = require("../../../red/runtime-registry/index");
|
|
|
|
var typeRegistry = require("../../../red/runtime-registry/registry");
|
2015-11-09 12:29:48 +01:00
|
|
|
|
|
|
|
describe('nodes/registry/installer', function() {
|
|
|
|
|
2018-04-23 15:24:51 +02: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-11-09 12:29:48 +01:00
|
|
|
before(function() {
|
2018-04-26 13:32:05 +02:00
|
|
|
installer.init({log:mockLog, settings:{}, events: new EventEmitter()});
|
2015-11-09 12:29:48 +01:00
|
|
|
});
|
2015-11-09 17:52:14 +01:00
|
|
|
afterEach(function() {
|
2018-01-24 23:56:54 +01:00
|
|
|
if (child_process.spawn.restore) {
|
|
|
|
child_process.spawn.restore();
|
|
|
|
}
|
2015-11-09 17:52:14 +01:00
|
|
|
if (child_process.execFile.restore) {
|
|
|
|
child_process.execFile.restore();
|
|
|
|
}
|
2015-11-14 21:27:04 +01:00
|
|
|
if (registry.addModule.restore) {
|
|
|
|
registry.addModule.restore();
|
|
|
|
}
|
|
|
|
if (registry.removeModule.restore) {
|
|
|
|
registry.removeModule.restore();
|
|
|
|
}
|
|
|
|
if (typeRegistry.removeModule.restore) {
|
|
|
|
typeRegistry.removeModule.restore();
|
|
|
|
}
|
|
|
|
if (registry.getModuleInfo.restore) {
|
|
|
|
registry.getModuleInfo.restore();
|
|
|
|
}
|
2017-01-22 00:46:44 +01:00
|
|
|
if (typeRegistry.getModuleInfo.restore) {
|
|
|
|
typeRegistry.getModuleInfo.restore();
|
|
|
|
}
|
2015-11-14 21:27:04 +01:00
|
|
|
|
2015-11-16 12:31:55 +01:00
|
|
|
if (require('fs').statSync.restore) {
|
|
|
|
require('fs').statSync.restore();
|
2015-11-14 21:27:04 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
});
|
2015-11-09 12:29:48 +01:00
|
|
|
|
|
|
|
describe("installs module", function() {
|
|
|
|
it("rejects when npm returns a 404", function(done) {
|
2018-01-24 23:56:54 +01:00
|
|
|
sinon.stub(child_process,"spawn",function(cmd,args,opt) {
|
|
|
|
var ee = new EventEmitter();
|
|
|
|
ee.stdout = new EventEmitter();
|
|
|
|
ee.stderr = new EventEmitter();
|
|
|
|
setTimeout(function() {
|
|
|
|
ee.stderr.emit('data'," 404 this_wont_exist");
|
|
|
|
ee.emit('close',1);
|
|
|
|
},10)
|
|
|
|
return ee;
|
2015-11-09 12:29:48 +01:00
|
|
|
});
|
|
|
|
|
2018-04-24 16:01:49 +02:00
|
|
|
installer.installModule("this_wont_exist").catch(function(err) {
|
2018-04-23 15:24:51 +02:00
|
|
|
err.should.have.property("code",404);
|
2015-11-09 12:29:48 +01:00
|
|
|
done();
|
|
|
|
});
|
|
|
|
});
|
2017-01-22 00:46:44 +01:00
|
|
|
it("rejects when npm does not find specified version", function(done) {
|
2018-01-24 23:56:54 +01:00
|
|
|
sinon.stub(child_process,"spawn",function(cmd,args,opt) {
|
|
|
|
var ee = new EventEmitter();
|
|
|
|
ee.stdout = new EventEmitter();
|
|
|
|
ee.stderr = new EventEmitter();
|
|
|
|
setTimeout(function() {
|
|
|
|
ee.stderr.emit('data'," version not found: this_wont_exist@0.1.2");
|
|
|
|
ee.emit('close',1);
|
|
|
|
},10)
|
|
|
|
return ee;
|
2017-01-22 00:46:44 +01:00
|
|
|
});
|
|
|
|
sinon.stub(typeRegistry,"getModuleInfo", function() {
|
|
|
|
return {
|
|
|
|
version: "0.1.1"
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
2018-04-24 16:01:49 +02:00
|
|
|
installer.installModule("this_wont_exist","0.1.2").catch(function(err) {
|
2017-01-22 00:46:44 +01:00
|
|
|
err.code.should.be.eql(404);
|
|
|
|
done();
|
|
|
|
});
|
|
|
|
});
|
|
|
|
it("rejects when update requested to existing version", function(done) {
|
|
|
|
sinon.stub(typeRegistry,"getModuleInfo", function() {
|
|
|
|
return {
|
|
|
|
version: "0.1.1"
|
|
|
|
}
|
|
|
|
});
|
2018-04-24 16:01:49 +02:00
|
|
|
installer.installModule("this_wont_exist","0.1.1").catch(function(err) {
|
2017-01-22 00:46:44 +01:00
|
|
|
err.code.should.be.eql('module_already_loaded');
|
|
|
|
done();
|
|
|
|
});
|
|
|
|
});
|
2015-11-09 12:29:48 +01:00
|
|
|
it("rejects with generic error", function(done) {
|
2018-01-24 23:56:54 +01:00
|
|
|
sinon.stub(child_process,"spawn",function(cmd,args,opt,cb) {
|
|
|
|
var ee = new EventEmitter();
|
|
|
|
ee.stdout = new EventEmitter();
|
|
|
|
ee.stderr = new EventEmitter();
|
|
|
|
setTimeout(function() {
|
|
|
|
ee.stderr.emit('data'," kaboom!");
|
|
|
|
ee.emit('close',1);
|
|
|
|
},10)
|
|
|
|
return ee;
|
2015-11-09 12:29:48 +01:00
|
|
|
});
|
|
|
|
|
|
|
|
installer.installModule("this_wont_exist").then(function() {
|
|
|
|
done(new Error("Unexpected success"));
|
2018-04-24 16:01:49 +02:00
|
|
|
}).catch(function(err) {
|
2015-11-09 12:29:48 +01:00
|
|
|
done();
|
|
|
|
});
|
|
|
|
});
|
|
|
|
it("succeeds when module is found", function(done) {
|
|
|
|
var nodeInfo = {nodes:{module:"foo",types:["a"]}};
|
2018-01-24 23:56:54 +01:00
|
|
|
sinon.stub(child_process,"spawn",function(cmd,args,opt) {
|
|
|
|
var ee = new EventEmitter();
|
|
|
|
ee.stdout = new EventEmitter();
|
|
|
|
ee.stderr = new EventEmitter();
|
|
|
|
setTimeout(function() {
|
|
|
|
ee.emit('close',0);
|
|
|
|
},10)
|
|
|
|
return ee;
|
2015-11-09 12:29:48 +01:00
|
|
|
});
|
|
|
|
var addModule = sinon.stub(registry,"addModule",function(md) {
|
|
|
|
return when.resolve(nodeInfo);
|
|
|
|
});
|
|
|
|
|
|
|
|
installer.installModule("this_wont_exist").then(function(info) {
|
|
|
|
info.should.eql(nodeInfo);
|
|
|
|
// commsMessages.should.have.length(1);
|
|
|
|
// commsMessages[0].topic.should.equal("node/added");
|
|
|
|
// commsMessages[0].msg.should.eql(nodeInfo.nodes);
|
|
|
|
done();
|
2018-04-24 16:01:49 +02:00
|
|
|
}).catch(function(err) {
|
2015-11-09 12:29:48 +01:00
|
|
|
done(err);
|
|
|
|
});
|
|
|
|
});
|
2015-11-09 17:52:14 +01:00
|
|
|
it("rejects when non-existant path is provided", function(done) {
|
2017-03-10 14:59:44 +01:00
|
|
|
this.timeout(10000);
|
2018-04-26 13:32:05 +02:00
|
|
|
var resourcesDir = path.resolve(path.join(__dirname,"resources","local","TestNodeModule","node_modules","NonExistant"));
|
2015-11-09 17:52:14 +01:00
|
|
|
installer.installModule(resourcesDir).then(function() {
|
|
|
|
done(new Error("Unexpected success"));
|
2018-04-24 16:01:49 +02:00
|
|
|
}).catch(function(err) {
|
2017-03-06 16:28:23 +01:00
|
|
|
if (err.hasOwnProperty("code")) {
|
|
|
|
err.code.should.eql(404);
|
|
|
|
done();
|
|
|
|
}
|
|
|
|
else {
|
2017-03-06 18:40:09 +01:00
|
|
|
console.log("ERRROR::"+err.toString()+"::");
|
|
|
|
err.toString().should.eql("Error: Install failed");
|
2017-03-06 16:28:23 +01:00
|
|
|
done();
|
|
|
|
}
|
2015-11-09 17:52:14 +01:00
|
|
|
});
|
|
|
|
});
|
|
|
|
it("succeeds when path is valid node-red module", function(done) {
|
|
|
|
var nodeInfo = {nodes:{module:"foo",types:["a"]}};
|
|
|
|
var addModule = sinon.stub(registry,"addModule",function(md) {
|
|
|
|
return when.resolve(nodeInfo);
|
|
|
|
});
|
2018-04-26 13:32:05 +02:00
|
|
|
var resourcesDir = path.resolve(path.join(__dirname,"resources","local","TestNodeModule","node_modules","TestNodeModule"));
|
2018-01-24 23:56:54 +01:00
|
|
|
sinon.stub(child_process,"spawn",function(cmd,args,opt) {
|
|
|
|
var ee = new EventEmitter();
|
|
|
|
ee.stdout = new EventEmitter();
|
|
|
|
ee.stderr = new EventEmitter();
|
|
|
|
setTimeout(function() {
|
|
|
|
ee.emit('close',0);
|
|
|
|
},10)
|
|
|
|
return ee;
|
2015-11-09 17:52:14 +01:00
|
|
|
});
|
|
|
|
installer.installModule(resourcesDir).then(function(info) {
|
|
|
|
info.should.eql(nodeInfo);
|
|
|
|
done();
|
2018-04-24 16:01:49 +02:00
|
|
|
}).catch(function(err) {
|
2015-11-09 17:52:14 +01:00
|
|
|
done(err);
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
2015-11-09 12:29:48 +01:00
|
|
|
});
|
|
|
|
describe("uninstalls module", function() {
|
|
|
|
it("rejects invalid module names", function(done) {
|
|
|
|
var promises = [];
|
|
|
|
promises.push(installer.uninstallModule("this_wont_exist "));
|
|
|
|
promises.push(installer.uninstallModule("this_wont_exist;no_it_really_wont"));
|
|
|
|
when.settle(promises).then(function(results) {
|
|
|
|
results[0].state.should.be.eql("rejected");
|
|
|
|
results[1].state.should.be.eql("rejected");
|
|
|
|
done();
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
it("rejects with generic error", function(done) {
|
|
|
|
var nodeInfo = [{module:"foo",types:["a"]}];
|
|
|
|
var removeModule = sinon.stub(registry,"removeModule",function(md) {
|
|
|
|
return when.resolve(nodeInfo);
|
|
|
|
});
|
2015-11-09 17:52:14 +01:00
|
|
|
sinon.stub(child_process,"execFile",function(cmd,args,opt,cb) {
|
2015-11-09 12:29:48 +01:00
|
|
|
cb(new Error("test_error"),"","");
|
|
|
|
});
|
|
|
|
|
|
|
|
installer.uninstallModule("this_wont_exist").then(function() {
|
|
|
|
done(new Error("Unexpected success"));
|
2018-04-24 16:01:49 +02:00
|
|
|
}).catch(function(err) {
|
2015-11-09 12:29:48 +01:00
|
|
|
done();
|
|
|
|
});
|
|
|
|
});
|
|
|
|
it("succeeds when module is found", function(done) {
|
|
|
|
var nodeInfo = [{module:"foo",types:["a"]}];
|
|
|
|
var removeModule = sinon.stub(typeRegistry,"removeModule",function(md) {
|
|
|
|
return nodeInfo;
|
|
|
|
});
|
|
|
|
var getModuleInfo = sinon.stub(registry,"getModuleInfo",function(md) {
|
|
|
|
return {nodes:[]};
|
|
|
|
});
|
2015-11-09 17:52:14 +01:00
|
|
|
sinon.stub(child_process,"execFile",function(cmd,args,opt,cb) {
|
2015-11-09 12:29:48 +01:00
|
|
|
cb(null,"","");
|
|
|
|
});
|
|
|
|
|
2015-11-16 12:31:55 +01:00
|
|
|
sinon.stub(fs,"statSync", function(fn) { return {}; });
|
2015-11-09 12:29:48 +01:00
|
|
|
|
|
|
|
installer.uninstallModule("this_wont_exist").then(function(info) {
|
|
|
|
info.should.eql(nodeInfo);
|
|
|
|
// commsMessages.should.have.length(1);
|
|
|
|
// commsMessages[0].topic.should.equal("node/removed");
|
|
|
|
// commsMessages[0].msg.should.eql(nodeInfo);
|
|
|
|
done();
|
2018-04-24 16:01:49 +02:00
|
|
|
}).catch(function(err) {
|
2015-11-09 12:29:48 +01:00
|
|
|
done(err);
|
|
|
|
});
|
|
|
|
});
|
|
|
|
});
|
|
|
|
});
|