介绍
拦截器是用 @Injectable()
装饰器注释并实现 NestInterceptor
接口的类。
拦截器具有一组有用的功能,这些功能的灵感来自 面向方面编程 (AOP) 技术。它们可以:
- 在方法执行之前/之后绑定额外的逻辑
- 转换函数返回的结果
- 转换函数抛出的异常
- 扩展基本功能行为
- 根据特定条件完全覆盖函数(例如,出于缓存目的)
在日常接口调试中,总会看到后端返回的数据格式都是一致的,例如:
1 2 3 4 5 6
| { "success": true, "message": "操作成功", "context": {}, "code": 200 }
|
接下来我会提到俩个案例,一个是接口返回拦截,一个是接口日志记录
接口调用记录
自定义元数据
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15
| import { SetMetadata } from "@nestjs/common"; import { LOG_KEY_METADATA } from "../constants/metadata.constant"; import { BusinessTypeEnum, LogProductEnum } from "../enum/log.enum";
export class LogOption { projectCode: LogProductEnum; title: string; businessType?: BusinessTypeEnum = BusinessTypeEnum.OTHER; } export const Log = (logOption: LogOption) => { return SetMetadata( LOG_KEY_METADATA, Object.assign(new LogOption(), logOption) ); };
|
表结构
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70
| import { Prop, Schema, SchemaFactory } from "@nestjs/mongoose"; import { DeviceSysCodeEnum } from "src/common/enum/device.enum"; import { BusinessTypeEnum, LogProductEnum, LogStatusEnum, } from "src/common/enum/log.enum";
export const LOG_MODEL_NAME = "db_logs";
@Schema() export class Log_Schema { @Prop() title: string;
@Prop({ default: BusinessTypeEnum.OTHER }) businessType: number;
@Prop({ default: DeviceSysCodeEnum.OTHER }) platformCode: string;
@Prop({ default: LogProductEnum.OTHER }) projectCode: number;
@Prop() method: string;
@Prop() requestParam: string;
@Prop() operateUser: string;
@Prop() operateUserId: string;
@Prop() ip: string;
@Prop() address: string;
@Prop({ default: LogStatusEnum.SUCCESS }) status: number;
@Prop() errorMsg: string;
@Prop() url: string;
@Prop() createTime: string; }
export const LogSchema = SchemaFactory.createForClass(Log_Schema);
|
涉及到的枚举
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49
| export enum LogProductEnum { OTHER = 0, ADMIN = 1, MERCHANT = 2, GLOBAL = 3, }
export enum BusinessTypeEnum { OTHER = 0, INSERT = 1, UPDATE = 2, DELETE = 3, SELECT = 4, GRANT = 5, EXPORT = 6, IMPORT = 7, }
export enum LogStatusEnum { SUCCESS = 0, FAIL = 1, }
export enum DeviceSysCodeEnum { H5 = "H5", PC = "PC", MOBILE = "MOBILE", MINI_PROGRAM = "MINI_PROGRAM", OTHER = "OTHER", }
|
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97
| import { CallHandler, ExecutionContext, Injectable, NestInterceptor, StreamableFile, } from "@nestjs/common"; import { Reflector } from "@nestjs/core"; import { Observable, tap } from "rxjs"; import { CommonService } from "src/shared/common/common.service"; import { LOG_KEY_METADATA } from "../constants/metadata.constant"; import { LogOption } from "../decorator/log.decorator"; import { HttpResult } from "../class/result.class"; import { DeviceSysCodeEnum } from "../enum/device.enum"; import { LogStatusEnum } from "../enum/log.enum"; import { AllExceptionFilter } from "../filter/all-exception.filter"; import { LogService } from "src/shared/log/log.service"; import { Log_Schema } from "src/shared/log/schemas/log.schema"; import { HttpStatusCode } from "../enum/http.enum";
@Injectable() export class LogInterceptor implements NestInterceptor { constructor( private readonly reflector: Reflector, private readonly commonService: CommonService, private readonly logService: LogService ) {} intercept(context: ExecutionContext, next: CallHandler): Observable<any> { return next.handle().pipe( tap({ next: data => { return this.log(context, data); }, error: error => { const allExceptionFilter = new AllExceptionFilter(); const { result } = allExceptionFilter.errorResult(error); return this.log(context, result); }, }) ); }
async log(context: ExecutionContext, data: HttpResult) { const logOption = this.reflector.get<LogOption>( LOG_KEY_METADATA, context.getHandler() ); if (!logOption) return true; const request = context.switchToHttp().getRequest(); const method = request.method.toUpperCase(); const logDto = new Log_Schema(); logDto.title = logOption.title; logDto.businessType = logOption.businessType; logDto.platformCode = request.headers["platformCode"] ?? DeviceSysCodeEnum.OTHER; logDto.projectCode = logOption.projectCode; logDto.method = method; logDto.requestParam = JSON.stringify({ params: request.params, query: request.query, body: request.body, }); logDto.url = request.url; logDto.operateUser = request?.user?.username ?? ""; logDto.operateUserId = request?.user?._id ?? ""; logDto.ip = this.commonService.getIp(request); logDto.address = await this.commonService.getLocation(logDto.ip); logDto.createTime = this.commonService.formatDate(); if ( (data && data.status === HttpStatusCode.OK) || data instanceof StreamableFile ) { logDto.status = LogStatusEnum.SUCCESS; } else { logDto.status = LogStatusEnum.FAIL; logDto.errorMsg = data && data.message; } await this.logService.create(logDto); } }
|
注册
在 app.module.ts
中:
1 2 3 4 5 6 7 8
| @Module({ providers: [ { provide: APP_INTERCEPTOR, useClass: LogInterceptor, }, ], })
|
控制层使用
1 2 3 4 5 6 7 8 9 10 11 12 13 14
| import { Controller, Post, Body, Get, Request } from "@nestjs/common"; export class UserController { constructor() {}
@Log({ title: "添加用户", projectCode: LogProductEnum.ADMIN, }) @Post("create") async create(@Body() body: { username: string; password: string }) { return "success"; } }
|
拦截器
拦截器代码
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26
| import { CallHandler, ExecutionContext, Injectable, NestInterceptor, } from "@nestjs/common"; import { map, Observable, tap } from "rxjs"; import { HttpResult } from "../class/result.class";
@Injectable() export class InterceptorInterceptor implements NestInterceptor { intercept(context: ExecutionContext, next: CallHandler): Observable<any> { const request = context.switchToHttp().getRequest(); return next.handle().pipe( tap(() => { console.log(request); }), map(data => { return HttpResult.success( data?.context ?? data, data?.message ?? "操作成功" ); }) ); } }
|
注册
在 app.module.ts
中:
1 2 3 4 5 6 7 8
| @Module({ providers: [ { provide: APP_INTERCEPTOR, useClass: InterceptorInterceptor, }, ], })
|
注意
LOG_KEY_METADATA
在这里
AllExceptionFilter
在这里
HttpResult
在这里