激情久久久_欧美视频区_成人av免费_不卡视频一二三区_欧美精品在欧美一区二区少妇_欧美一区二区三区的

服務器之家:專注于服務器技術及軟件下載分享
分類導航

node.js|vue.js|jquery|angularjs|React|json|js教程|

服務器之家 - 編程語言 - JavaScript - js教程 - 一個參數驗證,學會 Nest.js 的兩大機制:Pipe、ExceptionFilter

一個參數驗證,學會 Nest.js 的兩大機制:Pipe、ExceptionFilter

2021-12-28 23:29神光的編程秘籍神說要有光zxg js教程

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

一個參數驗證,學會 Nest.js 的兩大機制:Pipe、ExceptionFilter

本文轉載自微信公眾號「神光的編程秘籍」,作者神說要有光zxg。轉載本文請聯系神光的編程秘籍公眾號。

對輸入做驗證是一個 web 應用的基本功能,不止前端要做、后端也要做:

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

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

本文會學到這些內容:

  • Nest.js 的管道(pipe)做參數的驗證和轉換
  • Nest.js 的異常過濾器(exception filter)做異常的處理,返回響應
  • Nest.js 結合 class-validation 做聲明式的參數驗證

Nest.js 基礎

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

  • MVC 是 Controller、Service、Repository 的分層,這也是后端框架的通用架構
  • IOC 是依賴注入,也就是 Controller、Service、Repository 等實例都在 IOC 容器內可以自動注入,只需要聲明依賴,不需要手動 new。

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

整體架構如圖:

一個參數驗證,學會 Nest.js 的兩大機制:Pipe、ExceptionFilter

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

請求是在 Controller 里處理的,調用 Service 來完成業務邏輯,其中對數據庫的 CRUD 由 Repository 完成。

那么對參數的 validate 應該放在哪呢?

參數 validate 實現思路

對參數做驗證,在 Controller 里就可以,但是這種驗證邏輯是通用的,每個 Controller 里都做一遍也太麻煩了,能不能在 Controller 之前就做好了呢?

可能大家沒什么思路,那我們再了解一個 Nest.js 的功能:管道(Pipe)。

Nest.js 支持管道(Pipe),它會在請求到達 Controller 之前被調用,可以對參數做驗證和轉換,如果拋出了異常,則不會再傳遞給 Controller。

這種管道的特性適合用來做一些跨 Controller 的通用邏輯,比如 string 的 int 的轉換,參數驗證等等。

Nest.js 內置了 8 個管道:

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

可以分為 3 類:

parseXxx,把參數轉為某種類型;defaultValue,設置參數默認值;validation,做參數的驗證。

這些都是很通用的功能。

很明顯,validation 就可以用那個 ValidationPipe 來做。

但是我們先不著急用 Nest.js 提供的 Pipe,先自己實現下試試。

Pipe 的形式是實現 PipeTransform 接口的類,實現它的 transform 方法,在里面對 value 做各種轉換或者驗證,如果驗證失敗就拋一個異常。

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

之后我們在 IOC 容器啟動的時候調用 useGlobalPipes 方法注冊一下這個 Pipe:

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

我們來測試下:

當參數的 age 大于 20,就會拋異常返回對應的 response。

一個參數驗證,學會 Nest.js 的兩大機制:Pipe、ExceptionFilter

當參數小于 20,參數會被修改之后傳遞到 Controller:

一個參數驗證,學會 Nest.js 的兩大機制:Pipe、ExceptionFilter

一個參數驗證,學會 Nest.js 的兩大機制:Pipe、ExceptionFilter

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

這就是 Pipe 的作用。

所以,我們在 pipe 中對參數做 validate 就行了。可以用 class-validation 這個包,它支持裝飾器的方式來配置驗證規則:

類似這樣:

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

然后在 pipe 中調用 validate 的方法,如果有錯誤就拋異常:

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

因為我們是用裝飾器做的配置,那就要通過對象拿到它對應的類的裝飾器,所以在 validate 之前要調用 class-transformer 包的 plainToClass 方法來把普通的參數對象轉換為該類的實例。

這樣就實現了參數校驗的功能:

一個參數驗證,學會 Nest.js 的兩大機制:Pipe、ExceptionFilter

一個參數驗證,學會 Nest.js 的兩大機制:Pipe、ExceptionFilter

這就是 Nest.js 的 ValidationPipe 的實現原理。

當然,我們沒有做錯誤的格式化,不如內置 Pipe 做的漂亮,我們來看下內置 Pipe 的效果:

啟用內置的 ValidationPipe:

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

然后測試下:

一個參數驗證,學會 Nest.js 的兩大機制:Pipe、ExceptionFilter

人家這個返回的格式好多了。

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

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

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

異常過濾器的形式是一個實現 ExceptionFilter 接口的類,通過 Catch 裝飾器聲明對什么異常做處理。實現它的 catch 方法,在方法內拿到 response 對象返回相應的響應。

定義異常:

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

定義異常過濾器:

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

很明顯,之所以我們在 ValidationPipe 里只是拋了一個 BadRequestException 的錯誤,就返回了 400 的響應就是因為有內置的 ExceptionFilter。

Nest.js 內置了很多 ExceptionFilter,比如:

  • BadRequestException 返回 400,代表客戶端傳的參數有錯誤
  • ForbiddenException 返回 403,代表沒權限
  • NotFoundException 返回 404,代表沒找到資源

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

至此,我們實現了參數的 validate,通過 Pipe + ExceptionFilter。

總結

對輸入的驗證是一個基本功能,前后端都要做。

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

然后探究了 Nest.js 的 validate 的實現思路:驗證可以放在 Controller 之前,通過 Pipe 對參數做驗證和轉換,如果有錯誤就拋異常,異常會觸發 ExceptionFilter,從而返回不同的錯誤響應。

Pipe 在 Controller 之前被調用,如果拋出異常,請求就不會繼續傳遞到 Controller。

ExceptionFilter 可以監聽不同類型的 exception,做不同的響應。

內置有很多 Pipe 和 ExceptionFilter 可以直接用,不夠的時候還可以自己定義。

當然,如果只是實現驗證,不用這么麻煩,直接用 ValidationPipe 就行。

Validation 是一個基礎功能,但我們通過它學會了 Pipe 和 ExceptionFilter,還是很有意義的。

原文地址:https://mp.weixin.qq.com/s/MvYESCdTGnfegUn9WuSowA

延伸 · 閱讀

精彩推薦
  • js教程js實現頭像上傳并且可預覽提交

    js實現頭像上傳并且可預覽提交

    這篇文章主要介紹了js如何實現頭像上傳并且可預覽提交,幫助大家更好的理解和使用js,感興趣的朋友可以了解下...

    harold10244042021-12-20
  • js教程javascript的事件描述

    javascript的事件描述

    本文主要為大家介紹javascript事件的基礎知識,有需要的朋友可以參考下...

    js教程網9362021-12-15
  • js教程11個Javascript小技巧幫你提升代碼質量(小結)

    11個Javascript小技巧幫你提升代碼質量(小結)

    這篇文章主要介紹了11個Javascript小技巧幫你提升代碼質量(小結),文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需...

    alanyf6032021-12-21
  • js教程Jquery+javascript實現支付網頁數字鍵盤

    Jquery+javascript實現支付網頁數字鍵盤

    這篇文章主要為大家詳細介紹了Jquery+javascript實現支付網頁數字鍵盤,文中示例代碼介紹的非常詳細,具有一定的參考價值,感興趣的小伙伴們可以參考一...

    юноша 25702021-12-15
  • js教程js實現隨機點名功能

    js實現隨機點名功能

    這篇文章主要為大家詳細介紹了js實現隨機點名功能,文中示例代碼介紹的非常詳細,具有一定的參考價值,感興趣的小伙伴們可以參考一下...

    等待的L先生4892021-12-16
  • js教程js用正則表達式篩選年月日的實例方法

    js用正則表達式篩選年月日的實例方法

    在本篇文章里小編給大家整理的是一篇關于js用正則表達式篩選年月日的實例方法,對此有興趣的朋友們可以學習下。...

    小妮淺淺11842021-12-24
  • js教程Selenium執行JavaScript腳本的方法示例

    Selenium執行JavaScript腳本的方法示例

    這篇文章主要介紹了Selenium執行JavaScript腳本的方法示例,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友...

    測試開發小記6222021-12-23
  • js教程利用JavaScript為句子加標題的3種方法示例

    利用JavaScript為句子加標題的3種方法示例

    這篇文章主要給大家介紹了關于如何利用JavaScript為句子加標題的3種方法,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習...

    Hunter網絡安全10162021-12-27
主站蜘蛛池模板: 性欧美在线视频 | 久久av免费| chinese-xvideos| 久久国产中文 | 国产成人av一区 | 福利免费视频 | 毛片福利 | 92自拍视频 | 一区二区三级视频 | sesee99| 免费观看的毛片手机视频 | 日本逼逼视频 | 免费午夜视频在线观看 | 久久精品一区二区三区四区五区 | av在线大全 | 精品久久中文网址 | 国产精品美女久久久久久网站 | 国内成人自拍视频 | 精品成人免费一区二区在线播放 | 国产精品久久久久久影视 | 久久国产成人午夜av浪潮 | 亚洲第一成人在线观看 | 欧美一区黄 | 欧美国产日韩在线观看成人 | 欧洲亚洲激情 | 色综av| 国产毛片在线高清视频 | 一区二区久久精品66国产精品 | 色妇视频 | 国产精品久久久免费 | 9999久久久久久 | 国产成人av一区 | 91短视频在线观看 | 91精品国产一区二区在线观看 | 久久久久久中文字幕 | 男女无遮挡羞羞视频 | 免费在线观看亚洲 | 男女污污视频网站 | 国产精品一区二区三区在线看 | 91精品国产综合久久男男 | 在火车上摸两乳爽的大叫 |