第六步:加入日志功能

第一步:創(chuàng)建一個基礎項目
第二步:創(chuàng)建寫接口的模塊嘲玫,建立moogodb數(shù)據(jù)庫連接泽裳,寫添加與查詢接口
第三步:加入Swagger文檔
第四步:加入請求參數(shù)校驗
第五步:解決跨域問題网梢,返回 token,校驗token, 并定義將接收的參數(shù)全局存儲的方法

  • 注:不要關心注釋代碼畦幢,那是屬于后面功能的區(qū)域入热。因為隨著代碼體量加大,功能不再明確解总,只需按照步驟并參考效果圖贮匕,把關鍵代碼寫入即可,所以下只寫關鍵代碼花枫,具體請看效果圖粗合。
    項目地址

1 日志配置

// 日志 - 配置包描述
// 日志庫
import * as winston from 'winston';
// 日志庫的插件
import { WinstonModule } from 'nest-winston';
// 一個用于 winston 日志庫的插件 // https://www.npmjs.com/package/winston-daily-rotate-file
import * as DailyRotateFile from 'winston-daily-rotate-file';

mports: [
// 日志配置
    WinstonModule.forRoot({
      transports: [
        new DailyRotateFile({
          dirname: `logs`, // 日志保存的目錄
          // dirname: `D:\\`, // 日志保存的目錄 // 保存到本地
          filename: '%DATE%.log', // 日志名稱,%DATE% 占位符表示日期乌昔。隙疚。
          datePattern: 'YYYY-MM-DD', // 日志輪換的頻率,此處表示每天磕道。
          zippedArchive: true, // 是否通過壓縮的方式歸檔被輪換的日志文件供屉。
          maxSize: '20m', // 設置日志文件的最大大小,m 表示 mb 溺蕉。
          maxFiles: '14d', // 保留日志文件的最大天數(shù)伶丐,此處表示自動刪除超過 14 天的日志文件。
          // 記錄時添加時間戳信息
          format: winston.format.combine(
            winston.format.timestamp({
              format: 'YYYY-MM-DD HH:mm:ss',
            }),
            winston.format.json(),
          ),
        }),
      ],
    }),
  ]
效果圖

2 創(chuàng)建處理請求頭信息的方法

  • 在項目src文件夾下創(chuàng)建logServer文件夾疯特,并創(chuàng)建requestData.ts文件
~ /src/logServer/requestData.ts

// 處理請求頭信息的方法
import { Request } from 'express';

export const getReqMainInfo: (req: Request) => {
  [prop: string]: any;
} = (req) => {
  const { query, headers, url, method, body, connection } = req;
  
  // 獲取 IP
  const xRealIp = headers['X-Real-IP'];
  const xForwardedFor = headers['X-Forwarded-For'];
  const { ip: cIp } = req;
  const { remoteAddress } = connection || {};
  const ip = xRealIp || xForwardedFor || cIp || remoteAddress;

  return {
    url,
    host: headers.host,
    ip,
    method,
    query,
    body,
  };
};

3 開啟請求攔截器與異常過濾器來記錄日志

3.1 響應攔截器

3.1.1 在項目src文件夾下創(chuàng)建unify-response.interceptor.ts文件
~ /src/unify-response.interceptor.ts

// 在響應攔截器中記錄日志
// https://docs.nestjs.cn/9/interceptors
import {
  CallHandler,
  ExecutionContext,
  Inject,
  Injectable,
  NestInterceptor,
} from '@nestjs/common';
import { Observable } from 'rxjs';
import { map } from 'rxjs/operators';
import { Request } from 'express';
import { WINSTON_MODULE_PROVIDER } from 'nest-winston';
import { Logger } from 'winston';
import { getReqMainInfo } from './logServer/requestData';

@Injectable()
export class UnifyResponseInterceptor implements NestInterceptor {
  constructor(
    @Inject(WINSTON_MODULE_PROVIDER) private readonly logger: Logger,
  ) {}

  intercept(context: ExecutionContext, next: CallHandler): Observable<any> {
    const ctx = context.switchToHttp();
    const req = ctx.getRequest<Request>();

    return next.handle().pipe(
      map((data) => {
        this.logger.info('response', {
          responseData: data,
          req: getReqMainInfo(req),
        });
        
        return  data
      }),
    );
  }
}
3.1.2 在項目app.module.ts文件夾下引入并使用unify-response.interceptor.ts文件
~ /src/app.module.ts

//攔截器
import { APP_FILTER, APP_INTERCEPTOR } from '@nestjs/core';
// 全局響應攔截器
import { UnifyResponseInterceptor } from './unify-response.interceptor';

providers: [
    AppService,
    GlobalParamsService,
    // 應用響應攔截器
    {
      provide: APP_INTERCEPTOR,
      useClass: UnifyResponseInterceptor,
    },
  ],
效果圖

3.2 異常過濾器

3.2.1 在項目src文件夾下創(chuàng)建uinify-exception.filter.ts文件
~ /src/uinify-exception.filter.ts

// 在全局異常過濾器中記錄日志
// https://docs.nestjs.cn/9/exceptionfilters?id=%e5%bc%82%e5%b8%b8%e8%bf%87%e6%bb%a4%e5%99%a8-1
import {
  ArgumentsHost,
  Catch,
  ExceptionFilter,
  HttpException,
  HttpStatus,
  Inject,
} from '@nestjs/common';
import { Response, Request } from 'express';
import { WINSTON_MODULE_PROVIDER } from 'nest-winston';
import { Logger } from 'winston';
import { getReqMainInfo } from './logServer/requestData';

@Catch()
export default class UnifyExceptionFilter implements ExceptionFilter {
  // 注入日志服務相關依賴
  constructor(
    @Inject(WINSTON_MODULE_PROVIDER) private readonly logger: Logger,
  ) {}

  catch(exception: HttpException, host: ArgumentsHost) {
    const ctx = host.switchToHttp(); // 獲取當前執(zhí)行上下文
    const response  = ctx.getResponse<Response>(); // 獲取響應對象
    const request  = ctx.getRequest<Request>(); // 獲取請求對象
    const status =
      exception instanceof HttpException
        ? exception.getStatus()
        : HttpStatus.INTERNAL_SERVER_ERROR;

    const responses = response.status(status).json({
      statusCode: status,
      timestamp: new Date().toISOString(),
      path: request.url,
    });
    
    
    // 記錄日志(錯誤消息哗魂,錯誤碼,請求信息等)
    this.logger.error('', {
      status,
      req: getReqMainInfo(request),
      stack: exception.stack,
    });
    
    response.status(status >= 500 ? status : 201).json(response);
  }
}
3.1.2 在項目app.module.ts文件夾下引入并使用uinify-exception.filter.ts文件
~ /src/app.module.ts

//攔截器
import { APP_FILTER, APP_INTERCEPTOR } from '@nestjs/core';
// 全局異常過濾器
import UnifyExceptionFilter from './uinify-exception.filter';

providers: [
    AppService,
    GlobalParamsService,
   // 應用全局異常過濾器
    {
      provide: APP_FILTER,  
      useClass: UnifyExceptionFilter,
    },
  ],
效果圖

3 效果

  • 此時隨便調用一個接口就會在項目里看到日志
    效果圖
?著作權歸作者所有,轉載或內容合作請聯(lián)系作者
  • 序言:七十年代末漓雅,一起剝皮案震驚了整個濱河市录别,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌邻吞,老刑警劉巖组题,帶你破解...
    沈念sama閱讀 206,311評論 6 481
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場離奇詭異抱冷,居然都是意外死亡崔列,警方通過查閱死者的電腦和手機,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 88,339評論 2 382
  • 文/潘曉璐 我一進店門旺遮,熙熙樓的掌柜王于貴愁眉苦臉地迎上來赵讯,“玉大人盈咳,你說我怎么就攤上這事”咭恚” “怎么了鱼响?”我有些...
    開封第一講書人閱讀 152,671評論 0 342
  • 文/不壞的土叔 我叫張陵,是天一觀的道長讯私。 經(jīng)常有香客問我热押,道長,這世上最難降的妖魔是什么斤寇? 我笑而不...
    開封第一講書人閱讀 55,252評論 1 279
  • 正文 為了忘掉前任桶癣,我火速辦了婚禮,結果婚禮上娘锁,老公的妹妹穿的比我還像新娘牙寞。我一直安慰自己,他們只是感情好莫秆,可當我...
    茶點故事閱讀 64,253評論 5 371
  • 文/花漫 我一把揭開白布间雀。 她就那樣靜靜地躺著,像睡著了一般镊屎。 火紅的嫁衣襯著肌膚如雪惹挟。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 49,031評論 1 285
  • 那天缝驳,我揣著相機與錄音连锯,去河邊找鬼。 笑死用狱,一個胖子當著我的面吹牛运怖,可吹牛的內容都是我干的。 我是一名探鬼主播夏伊,決...
    沈念sama閱讀 38,340評論 3 399
  • 文/蒼蘭香墨 我猛地睜開眼摇展,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了溺忧?” 一聲冷哼從身側響起咏连,我...
    開封第一講書人閱讀 36,973評論 0 259
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎砸狞,沒想到半個月后捻勉,有當?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 43,466評論 1 300
  • 正文 獨居荒郊野嶺守林人離奇死亡刀森,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內容為張勛視角 年9月15日...
    茶點故事閱讀 35,937評論 2 323
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發(fā)現(xiàn)自己被綠了报账。 大學時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片研底。...
    茶點故事閱讀 38,039評論 1 333
  • 序言:一個原本活蹦亂跳的男人離奇死亡埠偿,死狀恐怖,靈堂內的尸體忽然破棺而出榜晦,到底是詐尸還是另有隱情冠蒋,我是刑警寧澤,帶...
    沈念sama閱讀 33,701評論 4 323
  • 正文 年R本政府宣布乾胶,位于F島的核電站抖剿,受9級特大地震影響,放射性物質發(fā)生泄漏识窿。R本人自食惡果不足惜斩郎,卻給世界環(huán)境...
    茶點故事閱讀 39,254評論 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望喻频。 院中可真熱鬧缩宜,春花似錦、人聲如沸甥温。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,259評論 0 19
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽姻蚓。三九已至宋梧,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間狰挡,已是汗流浹背捂龄。 一陣腳步聲響...
    開封第一講書人閱讀 31,485評論 1 262
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留圆兵,地道東北人跺讯。 一個月前我還...
    沈念sama閱讀 45,497評論 2 354
  • 正文 我出身青樓,卻偏偏與公主長得像殉农,于是被迫代替她去往敵國和親刀脏。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當晚...
    茶點故事閱讀 42,786評論 2 345

推薦閱讀更多精彩內容