loopback4-authentication
Version:
A loopback-next extension for authentication feature. Various Oauth strategies supported by this package.
74 lines (65 loc) • 2.47 kB
text/typescript
// Copyright IBM Corp. 2017,2018. All Rights Reserved.
// Node module: @loopback/authentication
// This file is licensed under the MIT License.
// License text available at https://opensource.org/licenses/MIT
import {AnyObject} from '@loopback/repository';
import {HttpErrors, Request, Response} from '@loopback/rest';
import {Strategy} from 'passport';
const passportRequestMixin = require('passport/lib/http/request');
/**
* Adapter class to invoke passport-strategy
* 1. provides express dependencies to the passport strategies
* 2. provides shimming of requests for passport authentication
* 3. provides lifecycle similar to express to the passport-strategy
* 3. provides state methods to the strategy instance
* see: https://github.com/jaredhanson/passport
*/
export class StrategyAdapter<T> {
/**
* @param strategy instance of a class which implements a passport-strategy;
* @description http://passportjs.org/
*/
constructor(private readonly strategy: Strategy) {}
/**
* The function to invoke the contained passport strategy.
* 1. Create an instance of the strategy
* 2. add success and failure state handlers
* 3. authenticate using the strategy
* @param request The incoming request.
*/
authenticate(
request: Request,
response?: Response,
options?: Object,
): Promise<T | void> {
return new Promise<T | void>((resolve, reject) => {
// mix-in passport additions like req.logIn and req.logOut
for (const key in passportRequestMixin) {
(request as AnyObject)[key] = passportRequestMixin[key];
}
// create a prototype chain of an instance of a passport strategy
const strategy = Object.create(this.strategy);
// add success state handler to strategy instance
strategy.success = (t: T) => {
resolve(t);
};
// add failure state handler to strategy instance
strategy.fail = (challenge: string) => {
reject(new HttpErrors.Unauthorized(challenge));
};
// add error state handler to strategy instance
strategy.error = (error: string) => {
reject(new HttpErrors.Unauthorized(error));
};
const REDIRECT_URL = 302;
strategy.redirect = (url: string) => {
if (response) {
response.redirect(REDIRECT_URL, url);
}
resolve();
};
// authenticate
strategy.authenticate(request, options ?? {});
});
}
}