0.0.6 • Published 4 years ago
@mikeevstropov/loopback4-acl v0.0.6
@mikeevstropov/loopback4-acl
A LoopBack 4 component for JWT based authorization support.
Installation
npm install --save loopback4-acl-componentBasic use
The following example shows the basic use of @acl.rules decorator in class and method levels.
@acl.rules([{
principal: ACLPrincipal.EVERYONE,
permission: ACLPermission.DENY,
}])
export class UserController {
@acl.rules([{
principal: ACLPrincipal.AUTHENTICATED,
permission: ACLPermission.ALLOW,
}])
@get('/users/whoAmI')
async whoAmI(): Promise<User> {
// ...
}
@acl.rules([{
principal: 'admin', // specific role
permission: ACLPermission.ALLOW,
}])
@get('/users/test')
async test() {
// ...
}
}From above the class level decorator denies access to all endpoints of
UserController for EVERYONE. But method level decorators allows the whoAmI method
for AUTHENTICATED and test method for
admin role.
How to make it work?
- Create your own
User(required) andRole(optional). - Implement
ACLUserServiceto resolve the session user. - Create
loginmethod to expose JWT token. - Mount
ACLComponentin your App.
Implement ACLUserService
The User Service is designed to resolve an instance of session user
by TokenPayload and its principals (roles) as option.
export class UserService implements ACLUserService {
constructor(
@repository(UserRepository)
public userRepository : UserRepository,
) {}
/**
* Resolve the Session User instance.
*/
public async resolveUser(tokenPayload: TokenPayload) {
return this.userRepository.findById(
tokenPayload.uid,
{include: ['roles']},
);
}
/**
* Resolve role-like names of the Session User.
*
* Optional.
* Do return an empty array if you're not using roles.
*/
public async resolvePrincipals(user: UserWithRelations) {
return user.roles.map((role: Role) => role.name);
}
}Create login method
It doesn't matter how you get the User instance in login
method, but you need to generate JWT token from its id.
export class UserController {
constructor(
@repository(UserRepository)
public userRepository: UserRepository,
@inject(ACLBindings.TOKEN_SERVICE)
private tokenService: ACLTokenService,
) {}
// ...
async login(
@requestBody(LoginRequestBody)
loginParameters: LoginParameters,
): Promise<LoginResponse> {
// ...
const user = await this.userRepository.findOne({
where: {username, password}
});
if (!user)
throw HttpErrors.Forbidden();
const token = await this.tokenService.encode(
{uid: user.id},
);
return token;
}
}Mount ACLComponent
Finally, bind your own ACLUserService and mount
the ACLComponent to your application.ts
export class App extends BootMixin() {
// ...
this.bind(ACLBindings.USER_SERVICE).toClass(UserService);
this.component(ACLComponent);
}Tests
run npm test from the root folder.
License
MIT