Nest.js 实战 (十):使用 winston 打印和收集日志记录

本文涉及的产品
日志服务 SLS,月写入数据量 50GB 1个月
简介: 这篇文章介绍了在Nest服务中如何使用Winston记录日志。文章首先强调了日志记录在后台服务中的重要性,接着提到Nest默认的内部日志记录器,并指出可以通过@nestjs/common包中的Logger类来全面控制日志系统的行为。文章还提到,为了在生产环境中实现更高级的日志功能,可以使用如Winston之类的Node.js日志包。接下来,文章介绍了如何在Nest服务中使用Winston记录日志,包括安装相关依赖、创建winston配置文件以及实现简单的日志记录示例。最后,文章指出更高级的自定义日志功能需要读者自己去探索。

前言

日志记录在后台服务的重要性不言而喻,它可以帮助开发者调试和故障排查性能监控审计和安全监控和警报等。

Nest 附带一个默认的内部日志记录器实现,它在实例化过程中以及在一些不同的情况下使用,比如发生异常等等(例如系统记录)。这由 @nestjs/common 包中的 Logger 类实现。你可以全面控制如下的日志系统的行为:

  1. 完全禁用日志
  2. 指定日志系统详细水平(例如,展示错误,警告,调试信息等)
  3. 覆盖默认日志记录器的时间戳(例如使用 ISO8601 标准作为日期格式)
  4. 完全覆盖默认日志记录器
  5. 通过扩展自定义默认日志记录器
  6. 使用依赖注入来简化编写和测试你的应用

更多高级的日志功能,可以使用任何 Node.js 日志包,比如Winston,来生成一个完全自定义的生产环境水平的日志系统。

今天我们就看看在 Nest 服务中应该如何使用 Winston 记录日志。

Nest 控制台

我们先看一下 Nest 服务原生的控制台输出:
trl1r148puvqjeeeyydn8yc451zc98jr.png
在接口请求和执行 SQL 的时候,控制台并没有相应的输出信息,这不方便我们排查和调试。

我们需要在服务执行操作的时候,控制台应该输出信息:

  1. 执行 SQL 时,打印 SQL 日志
  2. 调用接口时,打印接口请求日志
  3. 将接口调用时的日志生成保存到指定文件夹中

打印 Prisma 日志

由于我的项目是使用 Prisma 客户端,按照官网文档配置日志记录

PrismaService 中配置:

import {
   
    Injectable, OnModuleDestroy, OnModuleInit } from '@nestjs/common';
import {
   
    PrismaClient } from '@prisma/client';

@Injectable()
export class PrismaService extends PrismaClient implements OnModuleInit, OnModuleDestroy {
   
   
  constructor() {
   
   
    super({
   
   
      log: ['query', 'info', 'warn', 'error'], // 这里设置日志级别
    });
  }
  async onModuleInit() {
   
   
    await this.$connect(); // 在模块初始化时连接到数据库
  }

  async onModuleDestroy() {
   
   
    await this.$disconnect(); // 在应用程序关闭时断开与数据库的连
  }
}

在执行 SQL 时,控制台就会输出信息:
6cuibjxlbn3qpwmhgtnjtudhgu15udm9.png

接口请求日志

Nest 内部自带了 Logger 类,我们创建一个日志中间件:

import {
   
    Injectable, Logger, NestMiddleware } from '@nestjs/common';
import dayjs from 'dayjs';
import {
   
    NextFunction, Request, Response } from 'express';

@Injectable()
export class LoggerMiddleware implements NestMiddleware {
   
   
  private logger = new Logger();
  use(req: Request, res: Response, next: NextFunction) {
   
   
    // 记录开始时间
    const start = Date.now();
    // 获取请求信息
    const {
   
    method, originalUrl, ip, httpVersion, headers } = req;

    // 获取响应信息
    const {
   
    statusCode } = res;

    res.on('finish', () => {
   
   
      // 记录结束时间
      const end = Date.now();
      // 计算时间差
      const duration = end - start;

      // 这里可以根据自己需要组装日志信息:[timestamp] [method] [url] HTTP/[httpVersion] [client IP] [status code] [response time]ms [user-agent]
      const logFormat = `${
     
     dayjs().valueOf()} ${
     
     method} ${
     
     originalUrl} HTTP/${
     
     httpVersion} ${
     
     ip} ${
     
     statusCode} ${
     
     duration}ms ${
     
     headers['user-agent']}`;

      // 根据状态码,进行日志类型区分
      if (statusCode >= 500) {
   
   
        this.logger.error(logFormat, originalUrl);
      } else if (statusCode >= 400) {
   
   
        this.logger.warn(logFormat, originalUrl);
      } else {
   
   
        this.logger.log(logFormat, originalUrl);
      }
    });

    next();
  }
}

AppModule 中全局注册:

import {
   
    MiddlewareConsumer, Module, NestModule } from '@nestjs/common';

import {
   
    LoggerMiddleware } from '@/middleware/logger.middleware'; // 全局日志中间件

@Module({
   
   
  imports: [],
})
export class AppModule implements NestModule {
   
   
  configure(consumer: MiddlewareConsumer) {
   
   
    consumer.apply(LoggerMiddleware).forRoutes('*');
  }
}

在接口调用时,控制台就会输出信息:
7i5fk6lco9vsl1m1jfm5e4qstqlsxd09.png

Winston 生成日志

我们需要安装几个依赖:

  1. winston:一个通用的日志记录库,为 Node.js 应用提供灵活的日志记录功能
  2. nest-winston: 一个用于 winstonNest 模块包装器
  3. winston-daily-rotate-file: 用于将日志文件按天轮换保存
  4. chalk: 用于在终端中输出带有颜色的文本

终端执行命令:

pnpm add winston nest-winston winston-daily-rotate-file chalk@4

新建 winston 配置文件:

import chalk from 'chalk'; // 用于颜色化输出
import {
   
    createLogger, format, transports } from 'winston';
import DailyRotateFile from 'winston-daily-rotate-file';

// 定义日志级别颜色
const levelsColors = {
   
   
  error: 'red',
  warn: 'yellow',
  info: 'green',
  debug: 'blue',
  verbose: 'cyan',
};

const winstonLogger = createLogger({
   
   
  format: format.combine(format.timestamp(), format.errors({
   
    stack: true }), format.splat(), format.json()),
  defaultMeta: {
   
    service: 'log-service' },
  transports: [
    new DailyRotateFile({
   
   
      filename: 'logs/errors/error-%DATE%.log', // 日志名称,占位符 %DATE% 取值为 datePattern 值。
      datePattern: 'YYYY-MM-DD', // 日志轮换的频率,此处表示每天。
      zippedArchive: true, // 是否通过压缩的方式归档被轮换的日志文件。
      maxSize: '20m', // 设置日志文件的最大大小,m 表示 mb 。
      maxFiles: '14d', // 保留日志文件的最大天数,此处表示自动删除超过 14 天的日志文件。
      level: 'error', // 日志类型,此处表示只记录错误日志。
    }),
    new DailyRotateFile({
   
   
      filename: 'logs/warnings/warning-%DATE%.log',
      datePattern: 'YYYY-MM-DD',
      zippedArchive: true,
      maxSize: '20m',
      maxFiles: '14d',
      level: 'warn',
    }),
    new DailyRotateFile({
   
   
      filename: 'logs/app/app-%DATE%.log',
      datePattern: 'YYYY-MM-DD',
      zippedArchive: true,
      maxSize: '20m',
      maxFiles: '14d',
    }),
    new transports.Console({
   
   
      format: format.combine(
        format.colorize({
   
   
          colors: levelsColors,
        }),
        format.simple(),
        format.printf((info) => {
   
   
          // 获取 Info Symbols key
          const symbols = Object.getOwnPropertySymbols(info);
          const color = levelsColors[info[symbols[0]]]; // 获取日志级别的颜色
          const chalkColor = chalk[color];
          const message = `${
     
     chalkColor(info.timestamp)} ${
     
     chalkColor(info[symbols[2]])}`;
          return message;
        }),
      ),
      level: 'debug',
    }),
  ],
});

export default winstonLogger;

这里我们按照日志不同级别区分,在 AppModule 配置服务:

import {
   
    MiddlewareConsumer, Module, NestModule } from '@nestjs/common';
import {
   
    WinstonModule } from 'nest-winston';

import {
   
    LoggerMiddleware } from '@/middleware/logger.middleware'; // 全局日志中间件

import winstonLogger from './config/winston.config';

@Module({
   
   
  imports: [
    WinstonModule.forRoot({
   
   
      transports: winstonLogger.transports,
      format: winstonLogger.format,
      defaultMeta: winstonLogger.defaultMeta,
      exitOnError: false, // 防止意外退出
    }),
  ],
})
export class AppModule implements NestModule {
   
   
  configure(consumer: MiddlewareConsumer) {
   
   
    consumer.apply(LoggerMiddleware).forRoutes('*');
  }
}

main.ts 中更换日志记录器:

import {
   
    WINSTON_MODULE_NEST_PROVIDER } from 'nest-winston';

async function bootstrap() {
   
   
  const app = await NestFactory.create(AppModule);
  app.useLogger(app.get(WINSTON_MODULE_NEST_PROVIDER));
  await app.listen(3000);
}
bootstrap();

最终效果

j0whuhvn4re2lsz6po7lbmi0u3pvyr0i.png

总结

这里只是简单的日志记录示例,更加高级自定义的日志功能需要自己去探索。

Github 仓库: Vue3 Admin

相关实践学习
日志服务之使用Nginx模式采集日志
本文介绍如何通过日志服务控制台创建Nginx模式的Logtail配置快速采集Nginx日志并进行多维度分析。
相关文章
|
16天前
|
存储 安全 API
Next.js 实战 (九):使用 next-auth 完成第三方身份登录验证
这篇文章介绍了next-auth,一个为Next.js设计的身份验证库,支持多种认证方式,如电子邮件和密码、OAuth2.0提供商(如Google、GitHub、Facebook等)以及自定义提供商。文章包含了如何配置Github Provider以及会话管理,并提到了适配器Adapters在next-auth中的作用。最后,文章强调了next-auth的强大功能值得进一步探索。
53 10
|
26天前
|
设计模式 数据安全/隐私保护
Next.js 实战 (七):浅谈 Layout 布局的嵌套设计模式
这篇文章介绍了在Next.js框架下,如何处理中后台管理系统中特殊页面(如登录页)不包裹根布局(RootLayout)的问题。作者指出Next.js的设计理念是通过布局的嵌套来创建复杂的页面结构,这虽然保持了代码的整洁和可维护性,但对于特殊页面来说,却造成了不必要的布局包裹。文章提出了一个解决方案,即通过判断页面的skipGlobalLayout属性来决定是否包含RootLayout,从而实现特殊页面不包裹根布局的目标。
79 33
|
17天前
|
中间件 API
Next.js 实战 (八):使用 Lodash 打包构建产生的“坑”?
这篇文章介绍了作者在使用Nextjs15进行项目开发时遇到的部署问题。在部署过程中,作者遇到了打包构建时的一系列报错,报错内容涉及动态代码评估在Edge运行时不被允许等问题。经过一天的尝试和调整,作者最终删除了lodash-es库,并将radash的部分源码复制到本地,解决了打包报错的问题。文章最后提供了项目的线上预览地址,并欢迎读者留言讨论更好的解决方案。
23 10
|
2月前
|
前端开发 API 开发者
Next.js 实战 (五):添加路由 Transition 过渡效果和 Loading 动画
这篇文章介绍了Framer Motion,一个为React设计的动画库,提供了声明式API处理动画和页面转换,适合创建响应式用户界面。文章包括首屏加载动画、路由加载Loading、路由进场和退场动画等主题,并提供了使用Framer Motion和next.js实现这些动画的示例代码。最后,文章总结了这些效果,并邀请读者探讨更好的实现方案。
|
1月前
|
JavaScript 前端开发 API
Next.js 实战 (六):如何实现文件本地上传
这篇文章介绍了在Next.js中如何实现文件上传到本地的方法。文章首先提到Next.js官方文档中没有提供文件上传的实例代码,因此开发者需要自行实现,通常有两种思路:使用Node.js原生上传或使用第三方插件如multer。接着,文章选择了使用Node.js原生上传的方式来讲解实现过程,包括如何通过哈希值命名文件、上传到指定目录以及如何分类文件夹。然后,文章展示了具体的实现步骤,包括编写代码来处理文件上传,并给出了代码示例。最后,文章通过一个效果演示说明了如何通过postman模拟上传文件,并展示了上传后的文件夹结构。
|
12天前
|
监控 安全 中间件
Next.js 实战 (十):中间件的魅力,打造更快更安全的应用
这篇文章介绍了什么是Next.js中的中间件以及其应用场景。中间件可以用于处理每个传入请求,比如实现日志记录、身份验证、重定向、CORS配置等功能。文章还提供了一个身份验证中间件的示例代码,以及如何使用限流中间件来限制同一IP地址的请求次数。中间件相当于一个构建模块,能够简化HTTP请求的预处理和后处理,提高代码的可维护性,有助于创建快速、安全和用户友好的Web体验。
|
2月前
Next.js 实战 (二):搭建 Layouts 基础排版布局
本文介绍了作者在Next.js v15.x版本发布后,对一个旧项目的重构过程。文章详细说明了项目开发规范配置、UI组件库选择(最终选择了Ant-Design)、以及使用Ant Design的Layout组件实现中后台布局的方法。文末展示了布局的初步效果,并提供了GitHub仓库链接供读者参考学习。
Next.js 实战 (二):搭建 Layouts 基础排版布局
|
2月前
|
存储 网络架构
Next.js 实战 (四):i18n 国际化的最优方案实践
这篇文章介绍了Next.js国际化方案,作者对比了网上常见的方案并提出了自己的需求:不破坏应用程序的目录结构和路由。文章推荐使用next-intl库来实现国际化,并提供了详细的安装步骤和代码示例。作者实现了国际化切换时不改变路由,并把当前语言的key存储到浏览器cookie中,使得刷新浏览器后语言不会失效。最后,文章总结了这种国际化方案的优势,并提供Github仓库链接供读者参考。
|
2月前
Next.js 实战 (三):优雅的实现暗黑主题模式
这篇文章介绍了在Next.js中实现暗黑模式的具体步骤。首先,需要安装next-themes库。然后,在/components/ThemeProvider/index.tsx文件中新增ThemeProvider组件,并在/app/layout.tsx文件中注入该组件。如果想要加入过渡动画,可以修改代码实现主题切换时的动画效果。最后,需要在需要的位置引入ThemeModeButton组件,实现暗黑模式的切换。
|
3月前
|
设计模式 前端开发 JavaScript
JavaScript设计模式及其在实战中的应用,涵盖单例、工厂、观察者、装饰器和策略模式
本文深入探讨了JavaScript设计模式及其在实战中的应用,涵盖单例、工厂、观察者、装饰器和策略模式,结合电商网站案例,展示了设计模式如何提升代码的可维护性、扩展性和可读性,强调了其在前端开发中的重要性。
52 2