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.UnsignedInt
  name     String  @unique
  password String
}

model article {
  id       Int     @id @default(autoincrement()) @db.UnsignedInt
  title    String
  content  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请求,路由为register
    login(@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({ //执行配置好后的jwt
                name , 
                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        配置JWT
      JwtModule.registerAsync({
         imports: [ConfigModule],
         inject: [ConfigService],
         useFactory: (config: ConfigService) => {
             return {
               secret: config.get('TOKEN_SECRET'),    //密钥在.env里配置
               signOptions: { expiresIn: '60d' }    //持续时间
             }
         },
      })
   ],        //END
   controllers: [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) {
                   //TODO
                   return 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) {
                   //TODO
                   const prisma = new PrismaClient
                   const 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();

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mfbz.cn/a/571259.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

电商技术揭秘三十二:智能风控的案例研究与未来趋势

相关系列文章 电商技术揭秘相关系列文章合集&#xff08;1&#xff09; 电商技术揭秘相关系列文章合集&#xff08;2&#xff09; 电商技术揭秘二十八&#xff1a;安全与合规性保障 电商技术揭秘二十九&#xff1a;电商法律合规浅析 电商技术揭秘三十&#xff1a;知识产权保…

Excel 冻结前几行

Excel中有冻结首航和冻结首列的选项&#xff0c;但是如果想冻结前几行该怎么操作&#xff1f; 冻结首行或冻结首列 视图 -> 冻结窗格 -> 冻结首行或冻结首列 冻结前几行或前几列 视图 -> 冻结窗格 -> 冻结拆分窗格 具体冻结几行和几列取决于当前选中的单元格。…

leetcode 每日一题目 (树的直径 +DFS的深刻理解)

如下是题目的简单描述&#xff1a; 给你一棵二叉树的根节点 root &#xff0c;二叉树中节点的值 互不相同 。另给你一个整数 start 。在第 0 分钟&#xff0c;感染 将会从值为 start 的节点开始爆发。 每分钟&#xff0c;如果节点满足以下全部条件&#xff0c;就会被感染&…

2. 多机多卡运行nccl-tests对比分析

系列文章 第2章 多机多卡nccl-tests 对比分析 目录 系列文章前言一、本地环境1. 网卡接口2. RDMA3. TOPO信息pcie信息nvidia-smi topo -m 二、nccl-test对比分析1. 相关环境变量2. 不同情况的对比3. 总结与分析 前言 NCCL&#xff08;NVIDIA Collective Communications Libra…

带头双向循环链表的基本操作(c语言实现)

带头双向循环链表 带头双向循环链表是一种结合了双向链表和循环链表特性的数据结构。其主要特点如下&#xff1a; 双向性&#xff1a;链表中的每个节点都有两个指针&#xff0c;一个指向下一个节点&#xff08;next&#xff09;&#xff0c;另一个指向前一个节点&#xff08;p…

11.泛型

文章目录 1 泛型概念2. 自定义泛型结构3 泛型方法4 泛型在继承上的体现5 通配符的使用 1 泛型概念 所谓泛型就是用标识符标识不确定的类型&#xff0c;详细说就是&#xff1a;定义类或接口时用标识符表示类中某个属性的类型或者是某个方法的返回值及参数类型。泛型将在使用时&a…

《QT实用小工具·三十九》仿 Windows10 画图3D 的颜色选择器, 但更加强大

1、概述 源码放在文章末尾 该项目实现了仿 Windows10 画图3D 的颜色选择器&#xff0c;功能更加丰富更加强大。 项目部分代码如下所示&#xff1a; import QtQuick 2.15 import QtQuick.Controls 2.15 import QtQuick.Layouts 1.15 import QtGraphicalEffects 1.15Item {id…

基于OSAL 实现UART、LED、ADC等基础示例 4

1 UART 实验目的 串口在我们开发单片机项目是很重要的&#xff0c;可以观察我们的代码运行情况&#xff0c;本节的目的就 是实现串口双工收发。 虽然说 osal 相关的代码已经跟硬件关系不大了&#xff0c;但是我们还是来贴出相关的硬件原理图贴出来。 1.1 初始化 osal_init_s…

Leetcode743. 网络延迟时间

Every day a Leetcode 题目来源&#xff1a;743. 网络延迟时间 本题需要用到单源最短路径算法 Dijkstra&#xff0c;现在让我们回顾该算法&#xff0c;其主要思想是贪心。 将所有节点分成两类&#xff1a;已确定从起点到当前点的最短路长度的节点&#xff0c;以及未确定从起…

分类分析|KNN分类模型及其Python实现

KNN分类模型及其Python实现 1. KNN算法思想2. KNN算法步骤2.1 KNN主要优点2.2 KNN主要缺点 3. Python实现KNN分类算法3.1 自定义方法实现KNN分类3.2 调用scikit-learn模块实现KNN分类 4. K值的确定 在之前文章 分类分析|贝叶斯分类器及其Python实现中&#xff0c;我们对分类分…

DHCP的原理与配置

一.了解DHCP服务 1. DHCP (Dynamic Host Configuration Protocol)动态主机配置协议 是由Internet工作小组设计开发的&#xff0c;专门用于为TCP/IP网络中的计算机自动分配TCP/IP参数的协议 DHCP协议采用的是UDP作为传输协议&#xff0c;是给网络内的客户机自动分配IP地址&…

Redis入门到通关之Redis实现Session共享

文章目录 ☃️前期概要☃️基于Session实现登录方案☃️现有方案存在的问题☃️Redis代替Session的业务流程❄️❄️设计key的结构❄️❄️设计key的具体细节❄️❄️整体访问流程 欢迎来到 请回答1024 的博客 &#x1f353;&#x1f353;&#x1f353;欢迎来到 请回答1024的博…

羊大师分析,羊奶和牛奶哪个更适合夏天喝?

羊大师分析&#xff0c;羊奶和牛奶哪个更适合夏天喝&#xff1f; 羊奶和牛奶都是营养丰富的饮品&#xff0c;适合不同人群在不同季节饮用。在夏天&#xff0c;选择羊奶还是牛奶主要取决于个人的体质、口味偏好以及需求。 羊奶的营养价值较高&#xff0c;含有丰富的蛋白质、矿物…

ESP8266+STM32+阿里云保姆级教程(AT指令+MQTT)

前言&#xff1a;在开发过程中&#xff0c;几乎踩便了所有大坑小坑总结出的文章&#xff0c;我是把坑踩满了&#xff0c;帮助更过小白快速上手&#xff0c;如有错误之处&#xff0c;还麻烦各位大佬帮忙指正、 目录 一、ESP-01s介绍 1、ESP-01s管脚功能&#xff1a; 模组启动模…

元数据管理和数据目录对于现代数据平台的重要性——Lakehouse架构(四)

文章目录 前言解读元数据技术元数据业务元数据 元存储和数据目录如何协同工作&#xff1f;数据目录的特点查询、检索和发现数据数据分类数据治理数据血缘 前言 Lakehouse 架构中的存储层负责存储整个平台的数据&#xff0c;要查询存储的这些数据&#xff0c;我们需要一个数据目…

xgp怎么取消续费 微软商店xgp会员取消自动续费详细教程

xgp怎么取消续费 微软商店xgp会员取消自动续费详细教程 XGP这个游戏平台小伙伴们并不陌生吧&#xff0c;它是微软Xbox游戏部门推出的游戏租赁制会员服务&#xff0c;主要用于主机和PC两个平台。这个平台的会员就可以免费享受多款大制作游戏&#xff0c;而且每个月还会自动更新…

ruoyi-nbcio-plus基于vue3的flowable收回任务后重新进行提交表单的处理

更多ruoyi-nbcio功能请看演示系统 gitee源代码地址 前后端代码&#xff1a; https://gitee.com/nbacheng/ruoyi-nbcio 演示地址&#xff1a;RuoYi-Nbcio后台管理系统 http://122.227.135.243:9666/ 更多nbcio-boot功能请看演示系统 gitee源代码地址 后端代码&#xff1a…

JAVA毕业设计137—基于Java+Springboot+Vue的物流快递仓库管理系统(源代码+数据库)

毕设所有选题&#xff1a; https://blog.csdn.net/2303_76227485/article/details/131104075 基于JavaSpringbootVue的物流快递仓库管理系统(源代码数据库)137 一、系统介绍 本项目前后端分离&#xff0c;分为员工、销售员、仓库员、商品管理员、超级管理员五种角色 1、员工…

Linux 的情况下实现贪吃蛇 -- 第二十八天

1. 打印地图 keypad(stdsrc,1) 参数表示是否接收&#xff0c;1表示接收指令 2.思路&#xff1a;初始化initNcurses()&#xff0c; 封装地图函数实现地图gamePic&#xff08;&#xff09; 分三部分实现&#xff1a;2.1: 在第0行&#xff1a;打印 "--",&quo…

矩阵连乘算法

矩阵连乘&#xff1a; #include<iostream> #define inf 0x7fffffff using namespace std; int a[256] { 0 };//存储矩阵的行和列 int m[256][256] { 0 };//存储i到j的最少计算次数 int s[256][256] { 0 };//存储i到j的中转站k void m_print(int i, int j) {if (i …
最新文章