Autenticación y Registro con JWT en Nest.js: Implementación de un Sistema de Login y Register
En esta guía, aprenderás cómo implementar un sistema de autenticación robusto y seguro para tu aplicación Nest.js.
Utilizaremos JSON Web Tokens (JWT) para permitir que los usuarios se registren en tu plataforma y accedan a sus cuentas de manera segura. El proceso de login y register es fundamental para cualquier aplicación que requiera autenticación de usuarios. Con JWT, podremos generar tokens de acceso que protegerán nuestras rutas y garantizarán que solo los usuarios autenticados puedan acceder a determinados recursos.
Documentación oficial de Nest.js:
Código Github
Ayúdame a seguir creando contenido 😍
Tienes varias jugosas alternativas:
- Suscríbete al canal de Youtube (es gratis) click aquí
- Si estás viendo un video no olvides regalar un 👍 like y comentario 🙏🏼
- También puedes ser miembro del canal de Youtube click aquí
- Puedes adquirir cursos premium en Udemy 👇🏼👇🏼👇🏼
Muchas gracias por su tremendo apoyo 😊
Dependencias
bcryptjs
@nestjs/jwt
Users
En una aplicación Nest.js, los usuarios desempeñan un papel crucial en la autenticación y autorización. Aquí te mostramos cómo configurar la entidad de usuario y el módulo correspondiente para gestionar la información de los usuarios.
nest g res users --no-spec
Entidad de Usuario
Definiremos una entidad llamada User para representar la información de nuestros usuarios en la base de datos. Esta entidad contendrá campos importantes como id, name, email, password, rol, y deletedAt.
src\users\entities\user.entity.ts
import { Column, DeleteDateColumn, Entity } from "typeorm";
@Entity()
export class User {
@Column({ primary: true, generated: true })
id: number;
@Column({ length: 500 })
name: string;
@Column({ unique: true, nullable: false })
email: string;
@Column({ nullable: false })
password: string;
@Column({ default: "user" })
rol: string;
@DeleteDateColumn()
deletedAt: Date;
}
Módulo de Usuarios
Crearemos un módulo para gestionar los usuarios en nuestra aplicación. Utilizaremos el módulo TypeOrmModule para incluir nuestra entidad User y proporcionar acceso a la base de datos. Además, exportaremos el módulo TypeOrmModule para que podamos utilizarlo en otros módulos de nuestra aplicación.
src\users\users.module.ts
import { Module } from "@nestjs/common";
import { TypeOrmModule } from "@nestjs/typeorm";
import { User } from "./entities/user.entity";
import { UsersController } from "./users.controller";
import { UsersService } from "./users.service";
@Module({
imports: [TypeOrmModule.forFeature([User])],
controllers: [UsersController],
providers: [UsersService],
exports: [UsersService],
})
export class UsersModule {}
src\users\dto\create-user.dto.ts
export class CreateUserDto {
name: string;
email: string;
password: string;
}
src\users\users.service.ts
import { Injectable } from "@nestjs/common";
import { InjectRepository } from "@nestjs/typeorm";
import { Repository } from "typeorm";
import { CreateUserDto } from "./dto/create-user.dto";
import { UpdateUserDto } from "./dto/update-user.dto";
import { User } from "./entities/user.entity";
@Injectable()
export class UsersService {
constructor(
@InjectRepository(User)
private readonly usersRepository: Repository<User>
) {}
async create(createUserDto: CreateUserDto) {
return await this.usersRepository.save(createUserDto);
}
async findOneByEmail(email: string) {
return await this.usersRepository.findOneBy({ email });
}
}
Auth
yarn add bcryptjs
nest g module auth
nest g controller auth
nest g service auth
auth module
src\auth\auth.module.ts
import { Module } from "@nestjs/common";
import { UsersModule } from "src/users/users.module";
import { AuthController } from "./auth.controller";
import { AuthService } from "./auth.service";
@Module({
imports: [UsersModule],
controllers: [AuthController],
providers: [AuthService],
})
export class AuthModule {}
auth dto
src\auth\dto\register.dto.ts
import { Transform } from "class-transformer";
import { IsEmail, IsString, MinLength } from "class-validator";
export class RegisterDto {
@IsString()
@MinLength(1)
name: string;
@IsEmail()
email: string;
@IsString()
@MinLength(6)
@Transform(({ value }) => value.trim())
password: string;
}
src\auth\dto\login.dto.ts
import { Transform } from "class-transformer";
import { IsEmail, IsString, MinLength } from "class-validator";
export class LoginDto {
@IsEmail()
email: string;
@IsString()
@MinLength(6)
@Transform(({ value }) => value.trim())
password: string;
}
auth service
src\auth\auth.service.ts
import {
BadRequestException,
Injectable,
UnauthorizedException,
} from "@nestjs/common";
import { RegisterDto } from "./dto/register.dto";
import { JwtService } from "@nestjs/jwt";
import * as bcryptjs from "bcryptjs";
import { UsersService } from "src/users/users.service";
import { LoginDto } from "./dto/login.dto";
@Injectable()
export class AuthService {
constructor(private readonly usersService: UsersService) {}
async register({ password, email, name }: RegisterDto) {
const user = await this.usersService.findOneByEmail(email);
if (user) {
throw new BadRequestException("Email already exists");
}
const hashedPassword = await bcryptjs.hash(password, 10);
await this.usersService.create({
name,
email,
password: hashedPassword,
});
return {
message: "User created successfully",
};
}
async login({ email, password }: LoginDto) {
const user = await this.usersService.findOneByEmail(email);
if (!user) {
throw new UnauthorizedException("Invalid email");
}
const isPasswordValid = await bcryptjs.compare(password, user.password);
if (!isPasswordValid) {
throw new UnauthorizedException("Invalid password");
}
return {
email: user.email,
};
}
}
auth controller
src\auth\auth.controller.ts
import { Body, Controller, HttpCode, HttpStatus, Post } from "@nestjs/common";
import { LoginDto } from "./dto/login.dto";
import { RegisterDto } from "./dto/register.dto";
@Controller("auth")
export class AuthController {
constructor(private readonly authService: AuthService) {}
@Post("register")
register(@Body() registerDto: RegisterDto) {
return this.authService.register(registerDto);
}
@HttpCode(HttpStatus.OK)
@Post("login")
login(@Body() loginDto: LoginDto) {
return this.authService.login(loginDto);
}
}
@HttpCode(HttpStatus.OK)
El decorador @HttpCode(HttpStatus.OK)
es una característica de Nest.js que te permite establecer explícitamente el código de estado HTTP que se enviará como respuesta para una ruta específica de un controlador. En este caso particular, el decorador está configurado para devolver un código de estado HTTP 200 (OK) cuando se realiza una solicitud POST a la ruta /login.
JWT
yarn add @nestjs/jwt
src\auth\auth.service.ts
import {
BadRequestException,
Injectable,
UnauthorizedException,
} from "@nestjs/common";
import { RegisterDto } from "./dto/register.dto";
import { JwtService } from "@nestjs/jwt";
import * as bcryptjs from "bcryptjs";
import { UsersService } from "src/users/users.service";
import { LoginDto } from "./dto/login.dto";
@Injectable()
export class AuthService {
constructor(
private readonly usersService: UsersService,
private readonly jwtService: JwtService
) {}
async login({ email, password }: LoginDto) {
const user = await this.usersService.findOneByEmail(email);
if (!user) {
throw new UnauthorizedException("Invalid email");
}
const isPasswordValid = await bcryptjs.compare(password, user.password);
if (!isPasswordValid) {
throw new UnauthorizedException("Invalid password");
}
const payload = { email: user.email };
const token = await this.jwtService.signAsync(payload);
return {
token: token,
email: user.email,
};
}
}
src\auth\constants\jwt.constant.ts
export const jwtConstants = {
secret: "no utilizar en producción",
};
src\auth\auth.module.ts
import { Module } from "@nestjs/common";
import { JwtModule } from "@nestjs/jwt";
import { UsersModule } from "src/users/users.module";
import { AuthController } from "./auth.controller";
import { AuthService } from "./auth.service";
import { jwtConstants } from "./constants/jwt.constant";
@Module({
imports: [
UsersModule,
JwtModule.register({
global: true,
secret: jwtConstants.secret,
signOptions: { expiresIn: "1d" },
}),
],
controllers: [AuthController],
providers: [AuthService],
})
export class AuthModule {}
global
Estamos registrando el JwtModule
como global para facilitarnos las cosas. Esto significa que no necesitamos importar JwtModule
ningún otro lugar de nuestra aplicación.
Guard
En Nest.js, los guards son una funcionalidad que permite controlar si una determinada ruta o endpoint en una aplicación puede ser accedida por un cliente o no. Los guards se pueden utilizar para aplicar lógica de autenticación, autorización o cualquier otro tipo de validación antes de permitir que una solicitud llegue al controlador correspondiente.
nest g guard auth --no-spec
canActive
El método canActivate
es uno de los métodos principales que un guard puede implementar. Este método se ejecuta antes de que se procese una solicitud hacia una ruta o endpoint específico. La función canActivate
devuelve un valor booleano o una promesa que resuelve a un valor booleano. Dependiendo del resultado de esta función, se permitirá o se denegará el acceso al endpoint protegido.
Si el método canActivate
devuelve true, la solicitud se permite y continúa el flujo normal hacia el controlador asociado con la ruta.
Si el método canActivate
devuelve false, se niega el acceso a la ruta protegida y la solicitud se detiene. En este caso, el guard puede tomar acciones como redirigir a una página de inicio de sesión o enviar una respuesta de error, según el caso.
src\auth\guard\auth.guard.ts
import { CanActivate, ExecutionContext, Injectable } from "@nestjs/common";
import { JwtService } from "@nestjs/jwt";
import { Request } from "express";
@Injectable()
export class AuthGuard implements CanActivate {
constructor(private readonly jwtService: JwtService) {}
async canActivate(context: ExecutionContext): Promise<boolean> {
// El objeto context proporciona información
// sobre la solicitud entrante y el entorno de ejecución.
const request = context.switchToHttp().getRequest();
console.log(request.headers.authorization);
// Aquí puedes implementar tu lógica de autenticación o autorización.
// Por ejemplo, verificar si el usuario está autenticado, si tiene los roles adecuados, etc.
// Si la validación es exitosa, devuelve true, permitiendo el acceso.
// Si la validación falla, devuelve false, denegando el acceso.
return true; // o false, dependiendo de la lógica de tu guard.
}
private extractTokenFromHeader(request: Request) {
const [type, token] = request.headers.authorization?.split(" ") ?? [];
return type === "Bearer" ? token : undefined;
}
}
src\auth\auth.controller.ts
// import { Get, Request, UseGuards } from "@nestjs/common";
// import { AuthGuard } from "./guard/auth.guard";
Get('profile')
@UseGuards(AuthGuard)
profile(@Request() req) {
return 'profile';
}
src\auth\guard\auth.guard.ts
import {
CanActivate,
ExecutionContext,
Injectable,
UnauthorizedException,
} from "@nestjs/common";
import { JwtService } from "@nestjs/jwt";
import { Request } from "express";
import { jwtConstants } from "../constants/jwt.constant";
@Injectable()
export class AuthGuard implements CanActivate {
constructor(private readonly jwtService: JwtService) {}
async canActivate(context: ExecutionContext): Promise<boolean> {
const request = context.switchToHttp().getRequest();
const token = this.extractTokenFromHeader(request);
if (!token) {
throw new UnauthorizedException();
}
try {
const payload = await this.jwtService.verifyAsync(token, {
secret: jwtConstants.secret,
});
request.user = payload;
} catch (error) {
throw new UnauthorizedException();
}
return true;
}
private extractTokenFromHeader(request: Request) {
const [type, token] = request.headers.authorization?.split(" ") ?? [];
return type === "Bearer" ? token : undefined;
}
}
src\auth\auth.controller.ts
@Get('profile')
@UseGuards(AuthGuard)
profile(@Request() req) {
return req.user;
}