2014-11-06 22:59:48 +00:00
|
|
|
/**
|
2017-01-11 15:24:33 +00:00
|
|
|
* Copyright JS Foundation and other contributors, http://js.foundation
|
2014-11-06 22:59:48 +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-07-15 22:43:24 +01:00
|
|
|
|
2014-11-06 22:59:48 +00:00
|
|
|
var should = require("should");
|
|
|
|
var sinon = require("sinon");
|
|
|
|
|
|
|
|
var passport = require("passport");
|
|
|
|
|
2018-08-20 16:17:24 +01:00
|
|
|
var NR_TEST_UTILS = require("nr-test-utils");
|
|
|
|
|
|
|
|
var auth = NR_TEST_UTILS.require("@node-red/editor-api/lib/auth");
|
|
|
|
var Users = NR_TEST_UTILS.require("@node-red/editor-api/lib/auth/users");
|
|
|
|
var Tokens = NR_TEST_UTILS.require("@node-red/editor-api/lib/auth/tokens");
|
|
|
|
var Permissions = NR_TEST_UTILS.require("@node-red/editor-api/lib/auth/permissions");
|
2014-11-06 22:59:48 +00:00
|
|
|
|
2017-09-20 10:30:07 +01:00
|
|
|
describe("api/auth/index",function() {
|
2014-11-06 22:59:48 +00:00
|
|
|
|
|
|
|
|
2015-07-15 22:43:24 +01:00
|
|
|
|
2014-11-06 22:59:48 +00:00
|
|
|
describe("ensureClientSecret", function() {
|
2015-11-11 22:11:02 +00:00
|
|
|
before(function() {
|
2018-04-24 15:01:49 +01:00
|
|
|
auth.init({},{})
|
2015-11-11 22:11:02 +00:00
|
|
|
});
|
2014-11-06 22:59:48 +00:00
|
|
|
it("leaves client_secret alone if not present",function(done) {
|
|
|
|
var req = {
|
|
|
|
body: {
|
|
|
|
client_secret: "test_value"
|
|
|
|
}
|
|
|
|
};
|
|
|
|
auth.ensureClientSecret(req,null,function() {
|
|
|
|
req.body.should.have.a.property("client_secret","test_value");
|
|
|
|
done();
|
|
|
|
})
|
|
|
|
});
|
|
|
|
it("applies a default client_secret if not present",function(done) {
|
|
|
|
var req = {
|
|
|
|
body: { }
|
|
|
|
};
|
|
|
|
auth.ensureClientSecret(req,null,function() {
|
|
|
|
req.body.should.have.a.property("client_secret","not_available");
|
|
|
|
done();
|
|
|
|
})
|
|
|
|
});
|
|
|
|
});
|
2015-07-15 22:43:24 +01:00
|
|
|
|
2015-01-28 22:41:13 +00:00
|
|
|
describe("revoke", function() {
|
|
|
|
it("revokes a token", function(done) {
|
|
|
|
var revokeToken = sinon.stub(Tokens,"revoke",function() {
|
2020-11-30 14:38:48 +00:00
|
|
|
return Promise.resolve();
|
2015-01-28 22:41:13 +00:00
|
|
|
});
|
2015-07-15 22:43:24 +01:00
|
|
|
|
2015-01-28 22:41:13 +00:00
|
|
|
var req = { body: { token: "abcdef" } };
|
2015-07-15 22:43:24 +01:00
|
|
|
|
|
|
|
var res = { status: function(resp) {
|
2015-01-28 22:41:13 +00:00
|
|
|
revokeToken.restore();
|
|
|
|
|
|
|
|
resp.should.equal(200);
|
2015-10-03 20:32:24 +01:00
|
|
|
return {
|
|
|
|
end: done
|
|
|
|
}
|
2015-01-28 22:41:13 +00:00
|
|
|
}};
|
2015-07-15 22:43:24 +01:00
|
|
|
|
2015-01-28 22:41:13 +00:00
|
|
|
auth.revoke(req,res);
|
|
|
|
});
|
|
|
|
});
|
2015-07-15 22:43:24 +01:00
|
|
|
|
2015-01-28 22:41:13 +00:00
|
|
|
describe("login", function() {
|
2015-03-25 20:32:40 +00:00
|
|
|
beforeEach(function() {
|
|
|
|
sinon.stub(Tokens,"init",function(){});
|
|
|
|
sinon.stub(Users,"init",function(){});
|
|
|
|
});
|
|
|
|
afterEach(function() {
|
|
|
|
Tokens.init.restore();
|
|
|
|
Users.init.restore();
|
|
|
|
});
|
|
|
|
it("returns login details - credentials", function(done) {
|
2018-04-24 15:01:49 +01:00
|
|
|
auth.init({adminAuth:{type:"credentials"}},{})
|
2015-01-28 22:41:13 +00:00
|
|
|
auth.login(null,{json: function(resp) {
|
|
|
|
resp.should.have.a.property("type","credentials");
|
|
|
|
resp.should.have.a.property("prompts");
|
|
|
|
resp.prompts.should.have.a.lengthOf(2);
|
|
|
|
done();
|
|
|
|
}});
|
|
|
|
});
|
2015-03-25 20:32:40 +00:00
|
|
|
it("returns login details - none", function(done) {
|
2018-04-24 15:01:49 +01:00
|
|
|
auth.init({},{})
|
2015-03-25 20:32:40 +00:00
|
|
|
auth.login(null,{json: function(resp) {
|
|
|
|
resp.should.eql({});
|
|
|
|
done();
|
|
|
|
}});
|
|
|
|
});
|
2017-04-21 22:06:12 +01:00
|
|
|
it("returns login details - strategy", function(done) {
|
2018-04-24 15:01:49 +01:00
|
|
|
auth.init({adminAuth:{type:"strategy",strategy:{label:"test-strategy",icon:"test-icon"}}},{})
|
2017-04-12 10:09:03 +01:00
|
|
|
auth.login(null,{json: function(resp) {
|
2017-04-21 22:06:12 +01:00
|
|
|
resp.should.have.a.property("type","strategy");
|
2017-04-12 10:09:03 +01:00
|
|
|
resp.should.have.a.property("prompts");
|
|
|
|
resp.prompts.should.have.a.lengthOf(1);
|
|
|
|
resp.prompts[0].should.have.a.property("type","button");
|
2017-04-21 22:06:12 +01:00
|
|
|
resp.prompts[0].should.have.a.property("label","test-strategy");
|
2017-04-12 10:09:03 +01:00
|
|
|
resp.prompts[0].should.have.a.property("icon","test-icon");
|
|
|
|
|
|
|
|
done();
|
|
|
|
}});
|
|
|
|
});
|
2015-07-15 22:43:24 +01:00
|
|
|
|
2015-01-28 22:41:13 +00:00
|
|
|
});
|
2018-04-24 15:01:49 +01:00
|
|
|
describe("needsPermission", function() {
|
|
|
|
beforeEach(function() {
|
|
|
|
sinon.stub(Tokens,"init",function(){});
|
|
|
|
sinon.stub(Users,"init",function(){});
|
|
|
|
});
|
|
|
|
afterEach(function() {
|
|
|
|
Tokens.init.restore();
|
|
|
|
Users.init.restore();
|
|
|
|
if (passport.authenticate.restore) {
|
|
|
|
passport.authenticate.restore();
|
|
|
|
}
|
|
|
|
if (Permissions.hasPermission.restore) {
|
|
|
|
Permissions.hasPermission.restore();
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
|
|
it('no-ops if adminAuth not set', function(done) {
|
|
|
|
sinon.stub(passport,"authenticate",function(scopes,opts) {
|
|
|
|
return function(req,res,next) {
|
|
|
|
}
|
|
|
|
});
|
|
|
|
auth.init({});
|
|
|
|
var func = auth.needsPermission("foo");
|
|
|
|
func({},{},function() {
|
|
|
|
passport.authenticate.called.should.be.false();
|
|
|
|
done();
|
|
|
|
})
|
|
|
|
});
|
|
|
|
it('skips auth if req.user undefined', function(done) {
|
|
|
|
sinon.stub(passport,"authenticate",function(scopes,opts) {
|
|
|
|
return function(req,res,next) {
|
|
|
|
next();
|
|
|
|
}
|
|
|
|
});
|
|
|
|
sinon.stub(Permissions,"hasPermission",function(perm) { return true });
|
|
|
|
auth.init({adminAuth:{}});
|
|
|
|
var func = auth.needsPermission("foo");
|
|
|
|
func({user:null},{},function() {
|
|
|
|
try {
|
|
|
|
passport.authenticate.called.should.be.true();
|
|
|
|
Permissions.hasPermission.called.should.be.false();
|
|
|
|
done();
|
|
|
|
} catch(err) {
|
|
|
|
done(err);
|
|
|
|
}
|
|
|
|
})
|
|
|
|
});
|
|
|
|
|
|
|
|
it('passes for valid user permission', function(done) {
|
|
|
|
sinon.stub(passport,"authenticate",function(scopes,opts) {
|
|
|
|
return function(req,res,next) {
|
|
|
|
next();
|
|
|
|
}
|
|
|
|
});
|
|
|
|
sinon.stub(Permissions,"hasPermission",function(perm) { return true });
|
|
|
|
auth.init({adminAuth:{}});
|
|
|
|
var func = auth.needsPermission("foo");
|
|
|
|
func({user:true,authInfo: { scope: "read"}},{},function() {
|
|
|
|
try {
|
|
|
|
passport.authenticate.called.should.be.true();
|
|
|
|
Permissions.hasPermission.called.should.be.true();
|
|
|
|
Permissions.hasPermission.lastCall.args[0].should.eql("read");
|
|
|
|
Permissions.hasPermission.lastCall.args[1].should.eql("foo");
|
|
|
|
done();
|
|
|
|
} catch(err) {
|
|
|
|
done(err);
|
|
|
|
}
|
|
|
|
})
|
|
|
|
});
|
2015-07-15 22:43:24 +01:00
|
|
|
|
2018-04-24 15:01:49 +01:00
|
|
|
it('rejects for invalid user permission', function(done) {
|
|
|
|
sinon.stub(passport,"authenticate",function(scopes,opts) {
|
|
|
|
return function(req,res,next) {
|
|
|
|
next();
|
|
|
|
}
|
|
|
|
});
|
|
|
|
sinon.stub(Permissions,"hasPermission",function(perm) { return false });
|
|
|
|
auth.init({adminAuth:{}});
|
|
|
|
var func = auth.needsPermission("foo");
|
|
|
|
func({user:true,authInfo: { scope: "read"}},{
|
|
|
|
status: function(status) {
|
|
|
|
return { end: function() {
|
|
|
|
try {
|
|
|
|
status.should.eql(401);
|
|
|
|
done();
|
|
|
|
} catch(err) {
|
|
|
|
done(err);
|
|
|
|
}
|
|
|
|
}}
|
|
|
|
}
|
|
|
|
},function() {
|
|
|
|
done(new Error("hasPermission unexpected passed"))
|
|
|
|
});
|
|
|
|
});
|
|
|
|
});
|
2014-11-06 22:59:48 +00:00
|
|
|
});
|