Nest.js项目小结1

2024-04-25 05:12
文章标签 项目 js 小结 nest

本文主要是介绍Nest.js项目小结1,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

整体项目结构

        prisma\        (用于管理数据库迁移、定义数据模型和数据填充)

                migrations\        (日志记录,不用管)

                schema.prisma        (定义数据模型)

                seed.ts        (用于填充数据库初始数据的文件)

        src\

                auth\        (包含处理用户认证的模块代码)

                        dto\        (定义数据传输对象)

                                register.dto.ts        (数据对象)

                        auth.controller.ts         (处理路由等)

                        auth.module.ts            (模块)

                        auth.service.ts            (逻辑)

                common\        (包含通用的功能代码)

                        rules\        (自定义验证规则)

                                is-confirm.rule.ts

                                is-not-exists.rule.ts

                        validate.ts        (通用数据验证)

                prisma\

                        prisma.module.ts        (模块)

                        prisma.service.ts        (处理与Prisma数据库交互的服务)

                app.module.ts        (模块)

                main.ts        (主入口)

                transform.inteceptor.ts        (响应拦截器)

——————————————————————————————————————

schema.prisma       

        定义数据库模型

generator client {provider = "prisma-client-js"
}datasource db {provider = "mysql"url      = env("DATABASE_URL")
}model user {id       Int     @id @default(autoincrement()) @db.UnsignedIntname     String  @uniquepassword String
}model article {id       Int     @id @default(autoincrement()) @db.UnsignedInttitle    Stringcontent  String  @db.Text
}

seed.ts

        填充数据库数据

import { PrismaClient } from "@prisma/client"
import { hash } from "argon2"
import { Random } from "mockjs"const prisma = new PrismaClient()    //注册Prisma
async function run() {await prisma.user.create({    //往user表里面插入数据data: {name: "admin",password: await hash("admin8888")    //转成hash}})for (let i = 0; i < 50; i++) {    //填充50组await prisma.article.create({    //往article表里面插入数据data: {title: Random.ctitle(10, 30),    //随机题目content: Random.cparagraph(10, 20),    //随机内容}})}}
run()

——————————————————————————————————————

app.module.ts

import { Module } from '@nestjs/common';
import { AuthModule } from './auth/auth.module';    //引入验证模块
import { PrismaModule } from './prisma/prisma.module';    //引入prisma模块@Module({imports: [ AuthModule , PrismaModule ],    //插入
})
export class AppModule {}

——————————————————————————————————————

auth

        dto

                register.dto.ts

import { IsNotEmpty } from "class-validator";    //引入默认不为空验证
import { IsNotExistsRule } from "../../common/rules/is-not-exists.rule"; //引入自定义验证
import { IsConfiemRule } from "../../common/rules/is-confirm.rule";    //引入自定义验证export default class RegisterDto {    @IsNotEmpty({ message:'用户名不能为空' })                //验证规则@IsNotExistsRule("user" , { message: "用户已经存在" })   //验证规则name:string;                                             //类型@IsNotEmpty({ message:'密码不能为空' })@IsConfiemRule({ message: "两次密码不一致" })password:string@IsNotEmpty({message: "确认密码不能为空"})password_confirm: string
}

auth.controller.ts

import { Controller , Body , Post } from '@nestjs/common'
import { AuthService } from './auth.service';
import RegisterDto from './dto/register.dto';    //引入验证规则@Controller()
export class AuthController{constructor(private auth: AuthService) {}@Post('register')    //post请求,路由为registerlogin(@Body() dto: RegisterDto) {    //获取body数据 为dto : 验证规则return this.auth.register(dto)    //执行auth的register函数}
}

auth.service.ts

import { Injectable } from '@nestjs/common'
import RegisterDto from './dto/register.dto';
import { PrismaService } from './../prisma/prisma.service';    //引入全局配置prisma
import { hash } from 'argon2';
import { JwtService } from '@nestjs/jwt';@Injectable()
export class AuthService {constructor(private prisma:PrismaService , private jwt: JwtService){}async register(dto: RegisterDto){    //验证通过后执行这里const user = await this.prisma.user.create({    //等候创建data:{name: dto.name,password: await hash(dto.password)}})return this.token(user);    //执行token函数}private async token({ id , name }){return {token: await this.jwt.signAsync({ //执行配置好后的jwtname , sub: id })}}
}

auth.module.ts

import { Module } from '@nestjs/common'
import { AuthService } from './auth.service'
import { AuthController } from './auth.controller'
import { JwtModule } from '@nestjs/jwt'
import { ConfigModule, ConfigService } from '@nestjs/config'@Module({imports: [        //TODO        配置JWTJwtModule.registerAsync({imports: [ConfigModule],inject: [ConfigService],useFactory: (config: ConfigService) => {return {secret: config.get('TOKEN_SECRET'),    //密钥在.env里配置signOptions: { expiresIn: '60d' }    //持续时间}},})],        //ENDcontrollers: [AuthController],    //引入控制器providers: [AuthService]    //引入函数
})export class AuthModule{}

common

        rules

                is-confirm.rule.ts

import { PrismaClient } from '@prisma/client'
import { registerDecorator , ValidationArguments , ValidationOptions } from 'class-validator'//表字段是否唯一
export function IsConfiemRule(    //这里定义规则名称validationOptions?: ValidationOptions
) {return function (object: Record<string , any>, propertyName: string) {registerDecorator({name: 'IsConfiemRule',    //同上target: object.constructor,propertyName: propertyName,constraints: [],options: validationOptions,validator: {async validate(value: string, args: ValidationArguments) {//TODOreturn Boolean(value == args.object[`${args.property}_confirm`]) //true为通过,false为不通过//END}}})}
}

                is-not-exists.ule.ts

import { PrismaClient } from '@prisma/client'
import { registerDecorator , ValidationArguments , ValidationOptions } from 'class-validator'//表字段是否唯一
export function IsNotExistsRule(table: string,validationOptions?: ValidationOptions
) {return function (object: Record<string , any>, propertyName: string) {registerDecorator({name: 'IsNotExistsRule',target: object.constructor,propertyName: propertyName,constraints: [table],options: validationOptions,validator: {async validate(value: string, args: ValidationArguments) {//TODOconst prisma = new PrismaClientconst res = await prisma[table].findFirst({where: {[args.property]: value}})return !Boolean(res)//END}}})}
}
//同上

        validate.ts

import { HttpException, HttpStatus, ValidationPipe } from "@nestjs/common";
import { ValidationError } from "class-validator";export default class Validate extends ValidationPipe {protected flattenValidationErrors(validationErrors: ValidationError[]): string[] {const messages = {}validationErrors.forEach(error => {    //遍历出必要错误返回给前端messages[error.property] = Object.values(error.constraints)[0];})throw new HttpException({    //返回给前端的错误code: 422,    messages}, HttpStatus.UNPROCESSABLE_ENTITY); }
}

prisma

        prisma.module.ts

import { Global, Module } from '@nestjs/common'
import { PrismaService } from './prisma.service';@Global() // 全局模块
@Module({providers: [PrismaService],exports: [PrismaService]
})export class PrismaModule{}

        prisma.service.ts

import { PrismaClient } from '@prisma/client' 
import { Injectable } from '@nestjs/common'@Injectable()
export class PrismaService extends PrismaClient{constructor(){super({log:['query']})}
}

transform.inteceptor.ts

import { CallHandler , ExecutionContext , Injectable , NestInterceptor } from '@nestjs/common'
import { map } from 'rxjs/operators'@Injectable()
export class TransformInterceptor implements NestInterceptor {    //修改一下数据格式intercept(context: ExecutionContext, next: CallHandler) {return next.handle().pipe(map((data) => {return {data}}))}
}

main.ts

import { NestFactory } from '@nestjs/core';
import { AppModule } from './app.module';
import Validate from './common/validate';
import { TransformInterceptor } from './transform.inteceptor';async function bootstrap() {const app = await NestFactory.create(AppModule);app.useGlobalPipes(new Validate())    //返回给前端的错误数据app.useGlobalInterceptors(new TransformInterceptor())    //返回给前端的正确数据(响应拦截器)await app.listen(3000);    //监听的端口号
}
bootstrap();

这篇关于Nest.js项目小结1的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/933774

相关文章

Vite 打包目录结构自定义配置小结

《Vite打包目录结构自定义配置小结》在Vite工程开发中,默认打包后的dist目录资源常集中在asset目录下,不利于资源管理,本文基于Rollup配置原理,本文就来介绍一下通过Vite配置自定义... 目录一、实现原理二、具体配置步骤1. 基础配置文件2. 配置说明(1)js 资源分离(2)非 JS 资

Three.js构建一个 3D 商品展示空间完整实战项目

《Three.js构建一个3D商品展示空间完整实战项目》Three.js是一个强大的JavaScript库,专用于在Web浏览器中创建3D图形,:本文主要介绍Three.js构建一个3D商品展... 目录引言项目核心技术1. 项目架构与资源组织2. 多模型切换、交互热点绑定3. 移动端适配与帧率优化4. 可

sky-take-out项目中Redis的使用示例详解

《sky-take-out项目中Redis的使用示例详解》SpringCache是Spring的缓存抽象层,通过注解简化缓存管理,支持Redis等提供者,适用于方法结果缓存、更新和删除操作,但无法实现... 目录Spring Cache主要特性核心注解1.@Cacheable2.@CachePut3.@Ca

Java Stream 并行流简介、使用与注意事项小结

《JavaStream并行流简介、使用与注意事项小结》Java8并行流基于StreamAPI,利用多核CPU提升计算密集型任务效率,但需注意线程安全、顺序不确定及线程池管理,可通过自定义线程池与C... 目录1. 并行流简介​特点:​2. 并行流的简单使用​示例:并行流的基本使用​3. 配合自定义线程池​示

Java实现复杂查询优化的7个技巧小结

《Java实现复杂查询优化的7个技巧小结》在Java项目中,复杂查询是开发者面临的“硬骨头”,本文将通过7个实战技巧,结合代码示例和性能对比,手把手教你如何让复杂查询变得优雅,大家可以根据需求进行选择... 目录一、复杂查询的痛点:为何你的代码“又臭又长”1.1冗余变量与中间状态1.2重复查询与性能陷阱1.

Go之errors.New和fmt.Errorf 的区别小结

《Go之errors.New和fmt.Errorf的区别小结》本文主要介绍了Go之errors.New和fmt.Errorf的区别,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考... 目录error的基本用法1. 获取错误信息2. 在条件判断中使用基本区别1.函数签名2.使用场景详细对

SpringBoot通过main方法启动web项目实践

《SpringBoot通过main方法启动web项目实践》SpringBoot通过SpringApplication.run()启动Web项目,自动推断应用类型,加载初始化器与监听器,配置Spring... 目录1. 启动入口:SpringApplication.run()2. SpringApplicat

Springboot项目构建时各种依赖详细介绍与依赖关系说明详解

《Springboot项目构建时各种依赖详细介绍与依赖关系说明详解》SpringBoot通过spring-boot-dependencies统一依赖版本管理,spring-boot-starter-w... 目录一、spring-boot-dependencies1.简介2. 内容概览3.核心内容结构4.

在ASP.NET项目中如何使用C#生成二维码

《在ASP.NET项目中如何使用C#生成二维码》二维码(QRCode)已广泛应用于网址分享,支付链接等场景,本文将以ASP.NET为示例,演示如何实现输入文本/URL,生成二维码,在线显示与下载的完整... 目录创建前端页面(Index.cshtml)后端二维码生成逻辑(Index.cshtml.cs)总结

C#异步编程ConfigureAwait的使用小结

《C#异步编程ConfigureAwait的使用小结》本文介绍了异步编程在GUI和服务器端应用的优势,详细的介绍了async和await的关键作用,通过实例解析了在UI线程正确使用await.Conf... 异步编程是并发的一种形式,它有两大好处:对于面向终端用户的GUI程序,提高了响应能力对于服务器端应