偷偷摘套内射激情视频,久久精品99国产国产精,中文字幕无线乱码人妻,中文在线中文a,性爽19p

一個(gè)參數(shù)驗(yàn)證,學(xué)會(huì) Nest.js 的兩大機(jī)制:Pipe、ExceptionFilter

開發(fā) 前端
前端做表單的驗(yàn)證基本不用自己寫,有很多 validation 的庫,大家寫的也比較多了。后端的驗(yàn)證大家可能寫的相對(duì)較少,今天我們就來學(xué)下后端框架 Nest.js 如何做參數(shù)的驗(yàn)證吧。

[[442398]]

本文轉(zhuǎn)載自微信公眾號(hào)「神光的編程秘籍」,作者神說要有光zxg。轉(zhuǎn)載本文請(qǐng)聯(lián)系神光的編程秘籍公眾號(hào)。

對(duì)輸入做驗(yàn)證是一個(gè) web 應(yīng)用的基本功能,不止前端要做、后端也要做:

  • 前端做驗(yàn)證可以避免沒必要的請(qǐng)求,盡快給用戶反饋
  • 后端做驗(yàn)證可以防止一些繞過瀏覽器的惡意提交

前端做表單的驗(yàn)證基本不用自己寫,有很多 validation 的庫,大家寫的也比較多了。后端的驗(yàn)證大家可能寫的相對(duì)較少,今天我們就來學(xué)下后端框架 Nest.js 如何做參數(shù)的驗(yàn)證吧。

本文會(huì)學(xué)到這些內(nèi)容:

  • Nest.js 的管道(pipe)做參數(shù)的驗(yàn)證和轉(zhuǎn)換
  • Nest.js 的異常過濾器(exception filter)做異常的處理,返回響應(yīng)
  • Nest.js 結(jié)合 class-validation 做聲明式的參數(shù)驗(yàn)證

Nest.js 基礎(chǔ)

Nest.js 是基于 IOC 和 MVC 的思想的后端框架:

  • MVC 是 Controller、Service、Repository 的分層,這也是后端框架的通用架構(gòu)
  • IOC 是依賴注入,也就是 Controller、Service、Repository 等實(shí)例都在 IOC 容器內(nèi)可以自動(dòng)注入,只需要聲明依賴,不需要手動(dòng) new。

此外,Nest.js 還支持 Module,可以把 Controller、Service、Repository 封裝成一個(gè) Module,易于代碼的組織。

整體架構(gòu)如圖:

整個(gè) IOC 容器內(nèi)有多個(gè) Controller、Service、Respository 等實(shí)例,分散在不同的 Module 中。有一個(gè) AppModule 作為根來引入其他 Module。

請(qǐng)求是在 Controller 里處理的,調(diào)用 Service 來完成業(yè)務(wù)邏輯,其中對(duì)數(shù)據(jù)庫的 CRUD 由 Repository 完成。

那么對(duì)參數(shù)的 validate 應(yīng)該放在哪呢?

參數(shù) validate 實(shí)現(xiàn)思路

對(duì)參數(shù)做驗(yàn)證,在 Controller 里就可以,但是這種驗(yàn)證邏輯是通用的,每個(gè) Controller 里都做一遍也太麻煩了,能不能在 Controller 之前就做好了呢?

可能大家沒什么思路,那我們?cè)倭私庖粋€(gè) Nest.js 的功能:管道(Pipe)。

Nest.js 支持管道(Pipe),它會(huì)在請(qǐng)求到達(dá) Controller 之前被調(diào)用,可以對(duì)參數(shù)做驗(yàn)證和轉(zhuǎn)換,如果拋出了異常,則不會(huì)再傳遞給 Controller。

這種管道的特性適合用來做一些跨 Controller 的通用邏輯,比如 string 的 int 的轉(zhuǎn)換,參數(shù)驗(yàn)證等等。

Nest.js 內(nèi)置了 8 個(gè)管道:

  • ValidationPipe
  • ParseIntPipe
  • ParseBoolPipe
  • ParseArrayPipe
  • ParseUUIDPipe
  • ParseEnumPipe
  • ParseFloatPipe
  • DefaultValuePipe

可以分為 3 類:

parseXxx,把參數(shù)轉(zhuǎn)為某種類型;defaultValue,設(shè)置參數(shù)默認(rèn)值;validation,做參數(shù)的驗(yàn)證。

這些都是很通用的功能。

很明顯,validation 就可以用那個(gè) ValidationPipe 來做。

但是我們先不著急用 Nest.js 提供的 Pipe,先自己實(shí)現(xiàn)下試試。

Pipe 的形式是實(shí)現(xiàn) PipeTransform 接口的類,實(shí)現(xiàn)它的 transform 方法,在里面對(duì) value 做各種轉(zhuǎn)換或者驗(yàn)證,如果驗(yàn)證失敗就拋一個(gè)異常。

  1. import { PipeTransform, Injectable, ArgumentMetadata, BadRequestException } from '@nestjs/common'
  2.  
  3. @Injectable() 
  4. export class MyValidationPipe implements PipeTransform<any> { 
  5.   async transform(value: any, metadata: ArgumentMetadata) { 
  6.     if (value.age > 20) { 
  7.       throw new BadRequestException('年齡超過限制'); 
  8.     } else { 
  9.       value.age += 10; 
  10.     } 
  11.     return value; 
  12.   } 

之后我們?cè)?IOC 容器啟動(dòng)的時(shí)候調(diào)用 useGlobalPipes 方法注冊(cè)一下這個(gè) Pipe:

  1. import { NestFactory } from '@nestjs/core'
  2. import { AppModule } from './app.module'
  3. import { MyValidationPipe } from './pipes/MyValidationPipe'
  4.  
  5. async function bootstrap() { 
  6.   const app = await NestFactory.create(AppModule); 
  7.   app.useGlobalPipes(new MyValidationPipe()); 
  8.   await app.listen(3000); 
  9. bootstrap(); 

我們來測(cè)試下:

當(dāng)參數(shù)的 age 大于 20,就會(huì)拋異常返回對(duì)應(yīng)的 response。

當(dāng)參數(shù)小于 20,參數(shù)會(huì)被修改之后傳遞到 Controller:

可以看到,參數(shù)被傳遞到了 Controller 并且做了修改。

這就是 Pipe 的作用。

所以,我們?cè)?pipe 中對(duì)參數(shù)做 validate 就行了??梢杂?class-validation 這個(gè)包,它支持裝飾器的方式來配置驗(yàn)證規(guī)則:

類似這樣:

  1. import { IsEmail, IsNotEmpty, IsPhoneNumber, IsString } from "class-validator"
  2.  
  3. export class CreatePersonDto { 
  4.     @IsNotEmpty({ 
  5.         message: 'name 不能為空' 
  6.     }) 
  7.     @IsString() 
  8.     name: string; 
  9.  
  10.     @IsPhoneNumber("CN", { 
  11.         message: 'phone 不是一個(gè)電話號(hào)碼' 
  12.     }) 
  13.     phone: string; 
  14.  
  15.     @IsEmail({}, { 
  16.         message: 'email 不是一個(gè)合法郵箱' 
  17.     }) 
  18.     email: string; 

然后在 pipe 中調(diào)用 validate 的方法,如果有錯(cuò)誤就拋異常:

  1. import { PipeTransform, Injectable, ArgumentMetadata, BadRequestException } from '@nestjs/common'
  2. import { validate } from 'class-validator'
  3. import { plainToClass } from 'class-transformer'
  4.  
  5. @Injectable() 
  6. export class MyValidationPipe implements PipeTransform<any> { 
  7.   async transform(value: any, { metatype }: ArgumentMetadata) { 
  8.     if (!metatype) { 
  9.       return value; 
  10.     } 
  11.     const object = plainToClass(metatype, value); 
  12.     const errors = await validate(object); 
  13.     if (errors.length > 0) { 
  14.       throw new BadRequestException('Validation failed'); 
  15.     } 
  16.     return value; 
  17.   } 

因?yàn)槲覀兪怯醚b飾器做的配置,那就要通過對(duì)象拿到它對(duì)應(yīng)的類的裝飾器,所以在 validate 之前要調(diào)用 class-transformer 包的 plainToClass 方法來把普通的參數(shù)對(duì)象轉(zhuǎn)換為該類的實(shí)例。

這樣就實(shí)現(xiàn)了參數(shù)校驗(yàn)的功能:

這就是 Nest.js 的 ValidationPipe 的實(shí)現(xiàn)原理。

當(dāng)然,我們沒有做錯(cuò)誤的格式化,不如內(nèi)置 Pipe 做的漂亮,我們來看下內(nèi)置 Pipe 的效果:

啟用內(nèi)置的 ValidationPipe:

  1. import { ValidationPipe } from '@nestjs/common'
  2. import { NestFactory } from '@nestjs/core'
  3. import { AppModule } from './app.module'
  4.  
  5. async function bootstrap() { 
  6.   const app = await NestFactory.create(AppModule); 
  7.   app.useGlobalPipes(new ValidationPipe()); 
  8.   await app.listen(3000); 
  9. bootstrap(); 

然后測(cè)試下:

人家這個(gè)返回的格式好多了。

還有,大家有沒有注意到,我們只是返回了一個(gè) BadRequestException 的 error,但是服務(wù)器就返回了 400 的相應(yīng),這個(gè)是什么原因呢?

這就涉及到了 Nest.js 的另一個(gè)機(jī)制:異常過濾器(Exception Filter)。

Nest.js 支持異常過濾器(ExceptionFilter),可以聲明對(duì)什么錯(cuò)誤做什么響應(yīng),這樣應(yīng)用想返回什么響應(yīng)只需要拋相應(yīng)的異常。

異常過濾器的形式是一個(gè)實(shí)現(xiàn) ExceptionFilter 接口的類,通過 Catch 裝飾器聲明對(duì)什么異常做處理。實(shí)現(xiàn)它的 catch 方法,在方法內(nèi)拿到 response 對(duì)象返回相應(yīng)的響應(yīng)。

定義異常:

  1. export class ForbiddenException extends HttpException { 
  2.     constructor() { 
  3.         super('Forbidden', HttpStatus.FORBIDDEN); 
  4.     }  

定義異常過濾器:

  1. import { ExceptionFilter, Catch, ArgumentsHost, HttpException } from '@nestjs/common'
  2. import { Request, Response } from 'express'
  3.  
  4. @Catch(HttpException) 
  5. export class HttpExceptionFilter implements ExceptionFilter { 
  6.   catch(exception: HttpException, host: ArgumentsHost) { 
  7.     const ctx = host.switchToHttp(); 
  8.     const response = ctx.getResponse<Response>(); 
  9.     const request = ctx.getRequest<Request>(); 
  10.     const status = exception.getStatus(); 
  11.  
  12.     response 
  13.       .status(status) 
  14.       .json({ 
  15.         statusCode: status, 
  16.         timestamp: new Date().toISOString(), 
  17.         path: request.url, 
  18.       }); 
  19.   } 

很明顯,之所以我們?cè)?ValidationPipe 里只是拋了一個(gè) BadRequestException 的錯(cuò)誤,就返回了 400 的響應(yīng)就是因?yàn)橛袃?nèi)置的 ExceptionFilter。

Nest.js 內(nèi)置了很多 ExceptionFilter,比如:

  • BadRequestException 返回 400,代表客戶端傳的參數(shù)有錯(cuò)誤
  • ForbiddenException 返回 403,代表沒權(quán)限
  • NotFoundException 返回 404,代表沒找到資源

想返回什么響應(yīng)就拋什么 exception 就行,不夠的話還可以自定義 ExceptionFilter。

至此,我們實(shí)現(xiàn)了參數(shù)的 validate,通過 Pipe + ExceptionFilter。

總結(jié)

對(duì)輸入的驗(yàn)證是一個(gè)基本功能,前后端都要做。

我們先過了一下 Nest.js 的基礎(chǔ):Nest.js 是 MVC + IOC 的架構(gòu),并且支持 Module 來組織代碼。

然后探究了 Nest.js 的 validate 的實(shí)現(xiàn)思路:驗(yàn)證可以放在 Controller 之前,通過 Pipe 對(duì)參數(shù)做驗(yàn)證和轉(zhuǎn)換,如果有錯(cuò)誤就拋異常,異常會(huì)觸發(fā) ExceptionFilter,從而返回不同的錯(cuò)誤響應(yīng)。

Pipe 在 Controller 之前被調(diào)用,如果拋出異常,請(qǐng)求就不會(huì)繼續(xù)傳遞到 Controller。

ExceptionFilter 可以監(jiān)聽不同類型的 exception,做不同的響應(yīng)。

內(nèi)置有很多 Pipe 和 ExceptionFilter 可以直接用,不夠的時(shí)候還可以自己定義。

當(dāng)然,如果只是實(shí)現(xiàn)驗(yàn)證,不用這么麻煩,直接用 ValidationPipe 就行。

Validation 是一個(gè)基礎(chǔ)功能,但我們通過它學(xué)會(huì)了 Pipe 和 ExceptionFilter,還是很有意義的。

 

責(zé)任編輯:武曉燕 來源: 神光的編程秘籍
相關(guān)推薦

2021-06-18 06:48:54

前端Nest.js技術(shù)熱點(diǎn)

2022-02-02 20:21:24

短信驗(yàn)證碼登錄

2024-02-04 19:15:09

Nest.js管理項(xiàng)目

2021-10-28 17:40:22

Nest.js前端代碼

2022-03-18 21:51:10

Nest.jsAOP 架構(gòu)后端

2022-12-27 09:22:06

Nest.js框架

2021-12-22 06:56:06

MySQCrudjs

2022-03-02 14:00:46

Nest.jsExpress端口

2024-05-06 08:48:18

nestjava?MVC?

2011-06-21 15:42:32

筆記本技巧

2010-05-04 14:30:45

Oracle數(shù)據(jù)

2011-08-10 08:55:28

項(xiàng)目失敗

2024-05-21 10:35:34

2021-07-29 07:55:19

Demo 工作池

2009-11-30 16:55:10

微軟合作Novell

2011-11-02 09:35:34

虛擬化虛擬化管理

2013-09-09 11:14:30

2022-02-24 08:00:00

API混合云數(shù)據(jù)

2021-06-29 06:25:22

Nest.jsTypeORM數(shù)據(jù)庫

2011-07-01 10:42:51

IIS解析漏洞
點(diǎn)贊
收藏

51CTO技術(shù)棧公眾號(hào)