-
Notifications
You must be signed in to change notification settings - Fork 1
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
- Loading branch information
Showing
51 changed files
with
551 additions
and
171 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,7 @@ | ||
import { TokenPayload } from './src/shared/modules/auth/index.js'; | ||
|
||
declare module 'express-serve-static-core' { | ||
export interface Request { | ||
tokenPayload: TokenPayload; | ||
} | ||
} |
Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.
Oops, something went wrong.
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -7,4 +7,5 @@ export interface IRestSchema { | |
DB_PORT: string; | ||
DB_NAME: string; | ||
UPLOAD_DIRECTORY: string; | ||
JWT_SECRET: string; | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,49 @@ | ||
import { NextFunction, Request, Response } from 'express'; | ||
import { jwtVerify } from 'jose'; | ||
import { StatusCodes } from 'http-status-codes'; | ||
|
||
import { createSecretKey } from 'node:crypto'; | ||
|
||
import { Middleware } from './middleware.interface.js'; | ||
import { HttpError } from '../errors/index.js'; | ||
import { TokenPayload } from '../../../modules/auth/index.js'; | ||
|
||
function isTokenPayload(payload: unknown): payload is TokenPayload { | ||
return ( | ||
(typeof payload === 'object' && payload !== null) && | ||
('email' in payload && typeof payload.email === 'string') && | ||
('userName' in payload && typeof payload.userName === 'string') && | ||
('id' in payload && typeof payload.id === 'string') | ||
); | ||
} | ||
|
||
export class ParseTokenMiddleware implements Middleware { | ||
constructor(private readonly jwtSecret: string) {} | ||
|
||
public async execute(req: Request, _res: Response, next: NextFunction): Promise<void> { | ||
const authorizationHeader = req.headers?.authorization?.split(' '); | ||
if (!authorizationHeader) { | ||
return next(); | ||
} | ||
|
||
const [, token] = authorizationHeader; | ||
|
||
try { | ||
const { payload } = await jwtVerify(String(token), createSecretKey(this.jwtSecret, 'utf-8')); | ||
|
||
if (isTokenPayload(payload)) { | ||
req.tokenPayload = { ...payload }; | ||
return next(); | ||
} else { | ||
throw new Error('Bad token'); | ||
} | ||
} catch { | ||
|
||
return next(new HttpError( | ||
StatusCodes.UNAUTHORIZED, | ||
'Invalid token', | ||
'AuthenticateMiddleware') | ||
); | ||
} | ||
} | ||
} |
20 changes: 20 additions & 0 deletions
20
src/shared/libs/rest/middleware/private-route.middleware.ts
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,20 @@ | ||
|
||
import { StatusCodes } from 'http-status-codes'; | ||
import { NextFunction, Request, Response } from 'express'; | ||
|
||
import { Middleware } from './middleware.interface.js'; | ||
import { HttpError } from '../errors/index.js'; | ||
|
||
export class PrivateRouteMiddleware implements Middleware { | ||
public async execute({ tokenPayload }: Request, _: Response, next: NextFunction): Promise<void> { | ||
if (! tokenPayload) { | ||
throw new HttpError( | ||
StatusCodes.UNAUTHORIZED, | ||
'Unauthorized', | ||
'PrivateRouteMiddleware' | ||
); | ||
} | ||
|
||
return next(); | ||
} | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,7 @@ | ||
import { UserEntity } from '../../entities/index.js'; | ||
import { LoginUserDto } from '../user/index.js'; | ||
|
||
export interface AuthService { | ||
authenticate(user: UserEntity): Promise<string>; | ||
verify(dto: LoginUserDto): Promise<UserEntity>; | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,2 @@ | ||
export const JWT_ALGORITHM = 'HS256'; | ||
export const JWT_EXPIRED = '2d'; |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,15 @@ | ||
import { Container } from 'inversify'; | ||
|
||
import { AuthService } from './auth-service.interface.js'; | ||
import { DefaultAuthService } from './default-auth.service.js'; | ||
import { ExceptionFilter } from '../../libs/rest/index.js'; | ||
import { AuthExceptionFilter } from './auth.exception-filter.js'; | ||
import { COMPONENT } from '../../constants/component.constant.js'; | ||
|
||
export function createAuthContainer() { | ||
const authContainer = new Container(); | ||
authContainer.bind<AuthService>(COMPONENT.AUTH_SERVICE).to(DefaultAuthService).inSingletonScope(); | ||
authContainer.bind<ExceptionFilter>(COMPONENT.AUTH_EXCEPTION_FILTER).to(AuthExceptionFilter).inSingletonScope(); | ||
|
||
return authContainer; | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,29 @@ | ||
import { inject, injectable } from 'inversify'; | ||
import { NextFunction, Request, Response } from 'express'; | ||
|
||
import { ExceptionFilter } from '../../libs/rest/index.js'; | ||
import { Logger } from '../../libs/logger/index.js'; | ||
import { BaseUserException } from './errors/index.js'; | ||
import { COMPONENT } from '../../constants/component.constant.js'; | ||
|
||
@injectable() | ||
export class AuthExceptionFilter implements ExceptionFilter { | ||
constructor( | ||
@inject(COMPONENT.LOGGER) private readonly logger: Logger | ||
) { | ||
this.logger.info('Register AuthExceptionFilter'); | ||
} | ||
|
||
public catch(error: unknown, _: Request, res: Response, next: NextFunction): void { | ||
if (! (error instanceof BaseUserException)) { | ||
return next(error); | ||
} | ||
|
||
this.logger.error(`[AuthModule] ${error.message}`, error); | ||
res.status(error.httpStatusCode) | ||
.json({ | ||
type: 'AUTHORIZATION', | ||
error: error.message, | ||
}); | ||
} | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,56 @@ | ||
import { inject, injectable } from 'inversify'; | ||
import * as crypto from 'node:crypto'; | ||
import { SignJWT } from 'jose'; | ||
|
||
import { AuthService } from './auth-service.interface.js'; | ||
import { Logger } from '../../libs/logger/index.js'; | ||
import { LoginUserDto, UserService } from '../user/index.js'; | ||
import { TokenPayload } from './types/tokenPayload.js'; | ||
import { Config, IRestSchema } from '../../libs/config/index.js'; | ||
import { UserNotFoundException, UserPasswordIncorrectException } from './errors/index.js'; | ||
import { JWT_ALGORITHM, JWT_EXPIRED } from './auth.constant.js'; | ||
import { COMPONENT } from '../../constants/component.constant.js'; | ||
import { UserEntity } from '../../entities/index.js'; | ||
|
||
@injectable() | ||
export class DefaultAuthService implements AuthService { | ||
constructor( | ||
@inject(COMPONENT.LOGGER) private readonly logger: Logger, | ||
@inject(COMPONENT.USER_SERVICE) private readonly userService: UserService, | ||
@inject(COMPONENT.CONFIG) private readonly config: Config<IRestSchema>, | ||
) {} | ||
|
||
public async authenticate(user: UserEntity): Promise<string> { | ||
const jwtSecret = this.config.get('JWT_SECRET'); | ||
const secretKey = crypto.createSecretKey(jwtSecret, 'utf-8'); | ||
const tokenPayload: TokenPayload = { | ||
email: user.email, | ||
userName: user.userName, | ||
id: user.id, | ||
}; | ||
|
||
this.logger.info(`Create token for ${user.email}`); | ||
return new SignJWT(tokenPayload) | ||
.setProtectedHeader({ alg: JWT_ALGORITHM }) | ||
.setIssuedAt() | ||
.setExpirationTime(JWT_EXPIRED) | ||
.sign(secretKey); | ||
} | ||
|
||
public async verify(dto: LoginUserDto): Promise<UserEntity> { | ||
const user = await this.userService.findByEmail(dto.email); | ||
if (! user) { | ||
this.logger.warn(`User with ${dto.email} not found`); | ||
throw new UserNotFoundException(); | ||
} | ||
|
||
if (! user.verifyPassword(dto.password, this.config.get('SALT'))) { | ||
this.logger.warn(`Incorrect password for ${dto.email}`); | ||
throw new UserPasswordIncorrectException(); | ||
} | ||
|
||
return user; | ||
} | ||
|
||
|
||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,7 @@ | ||
import { HttpError } from '../../../libs/rest/index.js'; | ||
|
||
export class BaseUserException extends HttpError { | ||
constructor(httpStatusCode: number, message: string) { | ||
super(httpStatusCode, message); | ||
} | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,3 @@ | ||
export { BaseUserException } from './base-user.exception.js'; | ||
export { UserNotFoundException } from './user-not-found.exception.js'; | ||
export { UserPasswordIncorrectException } from './user-password-incorrect.exception.js'; |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,9 @@ | ||
import { StatusCodes } from 'http-status-codes'; | ||
|
||
import { BaseUserException } from './base-user.exception.js'; | ||
|
||
export class UserNotFoundException extends BaseUserException { | ||
constructor() { | ||
super(StatusCodes.NOT_FOUND, 'User not found'); | ||
} | ||
} |
9 changes: 9 additions & 0 deletions
9
src/shared/modules/auth/errors/user-password-incorrect.exception.ts
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,9 @@ | ||
import { StatusCodes } from 'http-status-codes'; | ||
|
||
import { BaseUserException } from './base-user.exception.js'; | ||
|
||
export class UserPasswordIncorrectException extends BaseUserException { | ||
constructor() { | ||
super(StatusCodes.UNAUTHORIZED, 'Incorrect user name or password'); | ||
} | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,4 @@ | ||
export { AuthService } from './auth-service.interface.js'; | ||
export { TokenPayload } from './types/tokenPayload.js'; | ||
export { createAuthContainer } from './auth.container.js'; | ||
export { DefaultAuthService } from './default-auth.service.js'; |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,5 @@ | ||
export type TokenPayload = { | ||
email: string; | ||
userName: string; | ||
id: string; | ||
}; |
Oops, something went wrong.