register needs to have a email verification code

This commit is contained in:
秦秋旭 2023-02-21 23:15:40 +08:00
parent 03f60a7059
commit 2db40172b2
11 changed files with 128 additions and 47 deletions

View File

@ -7,6 +7,11 @@
}, },
"compilerOptions": { "compilerOptions": {
"deleteOutDir": true, "deleteOutDir": true,
"plugins": ["@nestjs/swagger/plugin"] "plugins": [
{
"name": "@nestjs/swagger/plugin",
"options": { "introspectComments": true }
}
]
} }
} }

View File

@ -1,13 +1,12 @@
import { Module } from '@nestjs/common' import { Module } from '@nestjs/common'
import { AuthService } from './auth.service' import { AuthService } from './auth.service'
import { AuthController } from './auth.controller' import { AuthController } from './auth.controller'
import { UsersModule } from 'src/users/users.module' import { PasswordService } from './password.service'
import { JwtService } from '@nestjs/jwt' import { JwtService } from '@nestjs/jwt'
import { JwtStrategy } from './strategies/jwt.strategy' import { JwtStrategy } from './strategies/jwt.strategy'
@Module({ @Module({
controllers: [AuthController], controllers: [AuthController],
providers: [AuthService, JwtService, JwtStrategy], providers: [AuthService, JwtService, JwtStrategy, PasswordService],
imports: [UsersModule],
}) })
export class AuthModule {} export class AuthModule {}

View File

@ -1,32 +1,66 @@
import { import {
Inject, Inject,
Injectable, Injectable,
BadRequestException, ForbiddenException,
UnauthorizedException, UnauthorizedException,
} from '@nestjs/common' } from '@nestjs/common'
import { PasswordService } from 'src/users/password.service' import { PasswordService } from './password.service'
import { PrismaService } from 'nestjs-prisma'
import { Token, TokenPayload } from './dto/token.dto' import { Token, TokenPayload } from './dto/token.dto'
import { JwtService } from '@nestjs/jwt' import { JwtService } from '@nestjs/jwt'
import { securityConfig, SecurityConfig } from 'src/common/configs' import { securityConfig, SecurityConfig } from 'src/common/configs'
import { UsersService } from 'src/users/users.service'
import { CreateUserDto } from 'src/users/dto/create-user.dto' import { CreateUserDto } from 'src/users/dto/create-user.dto'
import { EmailSendDto } from 'src/email/dto/email.dto'
@Injectable() @Injectable()
export class AuthService { export class AuthService {
constructor( constructor(
private passwordService: PasswordService, private passwordService: PasswordService,
private jwtService: JwtService, private jwtService: JwtService,
private userService: UsersService, private prismaService: PrismaService,
@Inject(securityConfig.KEY) @Inject(securityConfig.KEY)
private secureConfig: SecurityConfig, private secureConfig: SecurityConfig,
) {} ) {}
async register(payload: CreateUserDto) { async register(userToCreate: CreateUserDto) {
const user = await this.userService.createUser(payload) console.log(userToCreate)
try {
const tokenPayload = this.jwtService.verify<EmailSendDto>(
userToCreate.token,
{
secret:
this.secureConfig.jwt_access_secret +
userToCreate.verificationCode +
'register',
},
)
if (
tokenPayload.email !== userToCreate.email ||
tokenPayload.scene !== 'register'
) {
throw new ForbiddenException('请输入正确的邮箱')
}
} catch (err) {
console.error(err)
throw new ForbiddenException(err.message)
}
const hashedPassword = await this.passwordService.hashPassword(
userToCreate.password,
)
const user = await this.prismaService.user.create({
data: {
username: userToCreate.username,
email: userToCreate.email,
password: hashedPassword,
},
})
return this.generateTokens({ userId: user.id }) return this.generateTokens({ userId: user.id })
} }
async login(email: string, password: string) { async login(email: string, password: string) {
const user = await this.userService.findUser({ email }) const user = await this.prismaService.user.findUniqueOrThrow({
where: { email },
})
const passwordValid = await this.passwordService.validatePassword( const passwordValid = await this.passwordService.validatePassword(
password, password,
@ -34,7 +68,7 @@ export class AuthService {
) )
if (!passwordValid) { if (!passwordValid) {
throw new BadRequestException('Invalid password') throw new ForbiddenException('Invalid password')
} }
return this.generateTokens({ userId: user.id }) return this.generateTokens({ userId: user.id })
@ -42,7 +76,7 @@ export class AuthService {
async refreshToken(token: string) { async refreshToken(token: string) {
try { try {
const { userId } = this.jwtService.verify(token, { const { userId } = this.jwtService.verify<TokenPayload>(token, {
secret: this.secureConfig.jwt_refresh_secret, secret: this.secureConfig.jwt_refresh_secret,
}) })
return this.generateTokens({ userId }) return this.generateTokens({ userId })

View File

@ -1,7 +1,17 @@
import { ApiProperty } from '@nestjs/swagger'
import { IsEmail, IsNotEmpty } from 'class-validator' import { IsEmail, IsNotEmpty } from 'class-validator'
export class EmailDto { export enum EmailScene {
register = 'register',
forgetPassword = 'forgetPassword',
}
export class EmailSendDto {
@IsNotEmpty() @IsNotEmpty()
@IsEmail() @IsEmail()
email: string email: string
@IsNotEmpty()
@ApiProperty({ enum: EmailScene, enumName: 'EmailScene' })
scene: EmailScene
} }

View File

@ -1,16 +1,22 @@
import { Body, Controller, Post } from '@nestjs/common' import { Body, Controller, Query, Post } from '@nestjs/common'
import { EmailService } from './email.service' import { EmailService } from './email.service'
import { ApiTags, ApiOperation } from '@nestjs/swagger' import { ApiTags, ApiOperation, ApiQuery } from '@nestjs/swagger'
import { EmailDto } from './dto/email.dto' import { EmailSendDto, EmailScene } from './dto/email.dto'
@ApiTags('Email') @ApiTags('Email')
@Controller('api/email') @Controller('api/email')
export class EmailController { export class EmailController {
constructor(private readonly emailService: EmailService) {} constructor(private readonly emailService: EmailService) {}
@ApiOperation({ summary: '发送邮件' }) // @ApiOperation({ summary: '测试邮件' })
@Post('test') // @Post('test')
async sendEmailTo(@Body() payload: EmailDto) { // async sendEmailTo(@Body() payload: EmailDto) {
return this.emailService.sendEmailTo(payload.email) // return this.emailService.sendEmailTo(payload.email)
// }
@ApiOperation({ summary: '发送邮箱验证码' })
@Post()
async getRegisterToken(@Body() payload: EmailSendDto) {
return this.emailService.getRegisterToken(payload.email, payload.scene)
} }
} }

View File

@ -1,4 +1,5 @@
import { Module } from '@nestjs/common' import { Module } from '@nestjs/common'
import { JwtService } from '@nestjs/jwt'
import { EmailService } from './email.service' import { EmailService } from './email.service'
import { EmailController } from './email.controller' import { EmailController } from './email.controller'
import { MailerModule } from '@nestjs-modules/mailer' import { MailerModule } from '@nestjs-modules/mailer'
@ -31,6 +32,6 @@ import { HandlebarsAdapter } from '@nestjs-modules/mailer/dist/adapters/handleba
}), }),
], ],
controllers: [EmailController], controllers: [EmailController],
providers: [EmailService], providers: [EmailService, JwtService],
}) })
export class EmailModule {} export class EmailModule {}

View File

@ -1,9 +1,20 @@
import { Injectable } from '@nestjs/common' import { ConflictException, Inject, Injectable } from '@nestjs/common'
import { securityConfig, SecurityConfig } from 'src/common/configs'
import { MailerService } from '@nestjs-modules/mailer' import { MailerService } from '@nestjs-modules/mailer'
import { JwtService } from '@nestjs/jwt'
import { EmailScene } from './dto/email.dto'
import { PrismaService } from 'nestjs-prisma'
@Injectable() @Injectable()
export class EmailService { export class EmailService {
constructor(private mailerService: MailerService) {} constructor(
private prismaService: PrismaService,
private mailerService: MailerService,
private jwtService: JwtService,
@Inject(securityConfig.KEY)
private secureConfig: SecurityConfig,
) {}
async sendEmailTo(email: string) { async sendEmailTo(email: string) {
return this.mailerService.sendMail({ return this.mailerService.sendMail({
to: email, // list of receivers to: email, // list of receivers
@ -16,4 +27,29 @@ export class EmailService {
}, },
}) })
} }
async getRegisterToken(email: string, scene: EmailScene) {
const user = await this.prismaService.user.findUnique({ where: { email } })
if (user) {
throw new ConflictException(`邮箱${email}已注册`)
}
const verificationCode = this.generateRandomNum()
const registerToken = this.jwtService.sign(
{ email, scene },
{
secret: this.secureConfig.jwt_access_secret + verificationCode + scene,
expiresIn: '30min',
},
)
await this.mailerService.sendMail({
to: email,
subject: '注册qiuxu.site',
html: `您正在注册qiuxu.site验证码为 <strong>${verificationCode}</strong>30分钟内有效`,
})
return { registerToken, verificationCode }
}
private generateRandomNum() {
return Math.floor(Math.random() * 899999 + 100000)
}
} }

View File

@ -18,4 +18,13 @@ export class CreateUserDto {
@IsNotEmpty() @IsNotEmpty()
@IsStrongPassword() @IsStrongPassword()
password: string password: string
/** @description 验证码 */
@IsNotEmpty()
@Length(6, 6)
verificationCode: string
/** @description 发送邮箱接口返回的Token */
@IsNotEmpty()
token: string
} }

View File

@ -1,11 +1,10 @@
import { Module } from '@nestjs/common' import { Module } from '@nestjs/common'
import { UsersService } from './users.service' import { UsersService } from './users.service'
import { UsersController } from './users.controller' import { UsersController } from './users.controller'
import { PasswordService } from './password.service'
@Module({ @Module({
controllers: [UsersController], controllers: [UsersController],
providers: [UsersService, PasswordService], providers: [UsersService],
exports: [UsersService, PasswordService], exports: [UsersService],
}) })
export class UsersModule {} export class UsersModule {}

View File

@ -1,19 +1,14 @@
import { Injectable, NotFoundException } from '@nestjs/common' import { Injectable, NotFoundException } from '@nestjs/common'
import { PrismaService } from 'nestjs-prisma' import { PrismaService } from 'nestjs-prisma'
import { CreateUserDto } from './dto/create-user.dto'
import { PasswordService } from './password.service'
import { Prisma } from '@prisma/client' import { Prisma } from '@prisma/client'
import { UserEntity } from './entities/user.entity' import { UserEntity } from './entities/user.entity'
@Injectable() @Injectable()
export class UsersService { export class UsersService {
constructor( constructor(private prismaService: PrismaService) {}
private prisma: PrismaService,
private passwordService: PasswordService,
) {}
async findUser(where: Prisma.UserWhereUniqueInput): Promise<UserEntity> { async findUser(where: Prisma.UserWhereUniqueInput): Promise<UserEntity> {
const user = await this.prisma.user.findUnique({ const user = await this.prismaService.user.findUnique({
where, where,
}) })
if (!user) { if (!user) {
@ -21,17 +16,4 @@ export class UsersService {
} }
return user return user
} }
async createUser(payload: CreateUserDto) {
const hashedPassword = await this.passwordService.hashPassword(
payload.password,
)
const user = await this.prisma.user.create({
data: {
...payload,
password: hashedPassword,
},
})
return user
}
} }