Nestjs使用log4j打印日志

众所周知,nest是自带日志的。但是好像没有log4j香,所以咱们来用log4j吧~

我只演示最简单的用法,用具体怎么样用大家可以自己进行封装。就像前端封装自己的请求一样。

一、安装

yarn add log4js stacktrace-js

二、使用

主要就三个文件:配置log4j文件、中间件文件、main.ts

配置log4j文件:src/utils/log4j.ts

import * as Path from 'path';
import * as Log4js from 'log4js';
import * as Util from 'util';
import * as Moment from 'moment'; // 处理时间的工具
import * as StackTrace from 'stacktrace-js';
import Chalk from 'chalk';
import config from '../config/log4js';// 日志级别
export enum LoggerLevel {ALL = 'ALL',MARK = 'MARK',TRACE = 'TRACE',DEBUG = 'DEBUG',INFO = 'INFO',WARN = 'WARN',ERROR = 'ERROR',FATAL = 'FATAL',OFF = 'OFF',
}// 内容跟踪类
export class ContextTrace {constructor(public readonly context: string,public readonly path?: string,public readonly lineNumber?: number,public readonly columnNumber?: number,) {}
}Log4js.addLayout('Awesome-nest', (logConfig: any) => {return (logEvent: Log4js.LoggingEvent): string => {let moduleName = '';let position = '';// 日志组装const messageList: string[] = [];logEvent.data.forEach((value: any) => {if (value instanceof ContextTrace) {moduleName = value.context;// 显示触发日志的坐标(行,列)if (value.lineNumber && value.columnNumber) {position = `${value.lineNumber}, ${value.columnNumber}`;}return;}if (typeof value !== 'string') {value = Util.inspect(value, false, 3, true);}messageList.push(value);});// 日志组成部分const messageOutput: string = messageList.join(' ');const positionOutput: string = position ? ` [${position}]` : '';const typeOutput = `[${logConfig.type}] ${logEvent.pid.toString()}   - `;const dateOutput = `${Moment(logEvent.startTime).format('YYYY-MM-DD HH:mm:ss',)}`;const moduleOutput: string = moduleName? `[${moduleName}] `: '[LoggerService] ';let levelOutput = `[${logEvent.level}] ${messageOutput}`;// 根据日志级别,用不同颜色区分switch (logEvent.level.toString()) {case LoggerLevel.DEBUG:levelOutput = Chalk.green(levelOutput);break;case LoggerLevel.INFO:levelOutput = Chalk.cyan(levelOutput);break;case LoggerLevel.WARN:levelOutput = Chalk.yellow(levelOutput);break;case LoggerLevel.ERROR:levelOutput = Chalk.red(levelOutput);break;case LoggerLevel.FATAL:levelOutput = Chalk.hex('#DD4C35')(levelOutput);break;default:levelOutput = Chalk.grey(levelOutput);break;}return `${Chalk.green(typeOutput)}${dateOutput}  ${Chalk.yellow(moduleOutput,)}${levelOutput}${positionOutput}`;};
});// 注入配置
Log4js.configure(config);// 实例化
const logger = Log4js.getLogger();
logger.level = LoggerLevel.TRACE;export class Logger {static trace(...args) {logger.trace(Logger.getStackTrace(), ...args);}static debug(...args) {logger.debug(Logger.getStackTrace(), ...args);}static log(...args) {logger.info(Logger.getStackTrace(), ...args);}static info(...args) {logger.info(Logger.getStackTrace(), ...args);}static warn(...args) {logger.warn(Logger.getStackTrace(), ...args);}static warning(...args) {logger.warn(Logger.getStackTrace(), ...args);}static error(...args) {logger.error(Logger.getStackTrace(), ...args);}static fatal(...args) {logger.fatal(Logger.getStackTrace(), ...args);}static access(...args) {const loggerCustom = Log4js.getLogger('http');loggerCustom.info(Logger.getStackTrace(), ...args);}// 日志追踪,可以追溯到哪个文件、第几行第几列static getStackTrace(deep = 2): string {const stackList: StackTrace.StackFrame[] = StackTrace.getSync();const stackInfo: StackTrace.StackFrame = stackList[deep];const lineNumber: number = stackInfo.lineNumber;const columnNumber: number = stackInfo.columnNumber;const fileName: string = stackInfo.fileName;const basename: string = Path.basename(fileName);return `${basename}(line: ${lineNumber}, column: ${columnNumber}): \n`;}
}

中间件文件:src/middleware/logger.middleware.ts

import { Request, Response } from 'express';
import { Logger } from '../utils/log4js';// 函数式中间件
export function logger(req: Request, res: Response, next: () => any) {const code = res.statusCode; // 响应状态码next();// 组装日志信息const logFormat = ` >>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>请求参数Request original url: ${req.originalUrl}Method: ${req.method}IP: ${req.ip}Status code: ${code}Parmas: ${JSON.stringify(req.params)}Query: ${JSON.stringify(req.query)}Body: ${JSON.stringify(req.body,)} \n  >>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>`;// 根据状态码,进行日志类型区分if (code >= 500) {Logger.error(logFormat);} else if (code >= 400) {Logger.warn(logFormat);} else {Logger.access(logFormat);Logger.log(logFormat);}
}

main.ts

import { logger } from './middleware/logger.middleware';async function bootstrap() {const app = await NestFactory.create(AppModule, { logger: console });app.use(logger);
}
bootstrap();

然后就可以记录日志辣~稍微详细一点的可以参考这位老哥的:https://blog.csdn.net/fwzzzzz/article/details/116160816

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/578274.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

ArkUI动画概述

目录 1、按照页面分类 2、按照功能分类 3、显示动画 4、属性动画 动画的原理是在一个时间段内,多次改变UI外观,由于人眼会产生视觉暂留,所以最终看到的就是一个“连续”的动画。UI的一次改变称为一个动画帧,对应一次屏幕刷新&a…

一开始我还不信!高德导航红绿灯竟然能读秒?

高德导航红绿灯为啥能读秒? 1 内部员工吐露 每天工作其实就是负责自己片区的红绿灯,一大早就去校对时间,然后发布到后台。是的,统计出来的,而且还是人工统计,有误差请见谅[害羞] 真的是很辛苦了&#xf…

npm常见错误

三个方面 1. npm ERR! code ELIFECYCLE npm ERR! errno 1 npm ERR! code ELIFECYCLE npm ERR! errno 1 npm ERR! phantomjs-prebuilt2.1.15 install: node install.js npm ERR! Exit status 1 npm ERR! npm ERR! Failed at the phantomjs-prebuilt2.1.15 install script. np…

如何使用队列处理 API 速率限制

对于遇到速率限制的应用程序来说也是一个挑战,因为它需要“放慢速度”或暂停。这是一个典型的场景: 初始请求:当应用程序发起与 API 的通信时,它会请求特定的数据或功能。API 响应: API 处理请求并响应请求的信息或执…

12.25

led.c #include "led.h" void all_led_init() {RCC_GPIO | (0X3<<4);//时钟使能GPIOE_MODER &(~(0X3<<20));//设置PE10输出GPIOE_MODER | (0X1<<20);//设置PE10为推挽输出GPIOE_OTYPER &(~(0x1<<10));//PE10为低速输出GPIOE_OSPEED…

单集群400TB,OceanBase稳定支撑快手核心业务场景

一款日均超过千万人访问的短视频 App 快手&#xff0c;面对高并发流量如何及时有效地处理用户请求&#xff1f;通过在后端配置多套 MySQL 集群来支撑高流量访问&#xff0c;以解决大数据量存储和性能问题&#xff0c;这种传统的 MySQL 分库分表方案有何问题&#xff1f;快手对分…

评估回馈电子负载的重要指标?

回馈电子负载是用于测试电源、电池和其他电子设备性能的设备。它可以模拟实际负载&#xff0c;同时将多余的能量回馈到电网或电池中。在选择和使用回馈电子负载时&#xff0c;有几个重要的指标需要考虑&#xff1a; 功率范围&#xff1a;回馈电子负载的功率范围是指其能够提供的…

巅峰画师Midjourney:新时代的独角兽

介绍 AI绘画领域中&#xff0c;Midjourney处于绝对地位&#xff0c;并且一年时间就登顶。 Midjourney是一家独立的AI研究实验室,探索新的思维媒介,拓展人类的想象力。 它由一个小型的自筹资金团队组成,专注于设计、人类基础设施和AI。 在AI绘画领域,Midjourney取得了非常突出…

百度Apollo五步入门自动驾驶:Dreamview与离线数据包分析(文末赠送apollo周边)

&#x1f3ac; 鸽芷咕&#xff1a;个人主页 &#x1f525; 个人专栏:《linux深造日志》《粉丝福利》 ⛺️生活的理想&#xff0c;就是为了理想的生活! ⛳️ 粉丝福利活动 ✅参与方式&#xff1a;通过连接报名观看课程&#xff0c;即可免费获取精美周边 ⛳️活动链接&#xf…

pytest 断言异常

一、前置说明 在 pytest 中,断言异常是通过 pytest 内置的 pytest.raises 上下文管理器来实现的。通过使用 pytest.raises,可以捕获并断言代码中引发的异常。 二、操作步骤 1. 编写测试代码 atme/demos/demo_pytest_tutorials/test_pytest_raises.py import pytest# 示例…

【PDF密码】 一键强制去掉pdf密码

想要给PDF文件设置一个密码防止他人对文件进行编辑&#xff0c;那么我们可以对PDF文件设置限制编辑&#xff0c;设置方法很简单&#xff0c;我们在PDF编辑器中点击文件 – 属性 – 安全&#xff0c;在权限下拉框中选中【密码保护】 然后在密码保护界面中&#xff0c;我们勾选【…

通过three.js玩转车展项目

1.项目搭建 1.1 创建文件夹 mkdir 文件名1.2 初始化package.json npm init -y1.3 安装打包工具并配置相关依赖 npm i parcel -d在package.json中打包路径和指令 1.4 安装three.js npm i three -d2.项目搭建 2.1 新建index.html&#xff0c;并再index.html引入car.js,在…

【es6】async、await原理

async、await是es6新出的&#xff0c;主要是为了解决多个promise函数产生的嵌套层级过多的问题。 async、await是基于generator实现的代码中断操作&#xff08;上一个await未处理完时&#xff0c;代码不会继续向下执行&#xff0c;看上去就是中断了代码&#xff09; generator…

Mysql数据库批量更新表编码及排序规则

SELECT CONCAT( ALTER TABLE , TABLE_NAME, DEFAULT CHARACTER SET utf8mb4 COLLATE utf8mb4_unicode_ci; ) AS 修正SQL FROM information_schema.TABLES WHERE TABLE_COLLATION ! utf8mb4_unicode_ci AND TABLE_SCHEMA edu-integrated; SELECT…

从流星雨启程:Python和Pygame下载与安装全过程

文章目录 一、前言二、下载安装过程1.官网下载安装包2.安装python过程第一步第二步第三步第四步第五步安装完成 3.简单测试Python3.1 检查 Python 版本号3.2 打开 Python 解释器3.3 输入你的第一个代码3.4 运行 Python 脚本 4.安装Pygame4.1 cmd命令安装Pygame4.2 pip升级4.3 安…

zookeeper 面试

1zookeeper 是什么&#xff1f; 是一个开源的分布式协调服务&#xff0c;它提供了一个具有高可用性和一致性的分布式环境&#xff0c;用于协调和管理分布式系统中的各种数据和状态。 2 zookeeper 都有哪些功能&#xff1f; 分布式锁&#xff1a;可以通过 ZooKeeper 实现分布式…

实战:朴素贝叶斯文本分类器搭建与性能评估

&#x1f497;&#x1f497;&#x1f497;欢迎来到我的博客&#xff0c;你将找到有关如何使用技术解决问题的文章&#xff0c;也会找到某个技术的学习路线。无论你是何种职业&#xff0c;我都希望我的博客对你有所帮助。最后不要忘记订阅我的博客以获取最新文章&#xff0c;也欢…

ROS程序中常用循环结构的用途和用法

在 ROS (Robot Operating System) 程序中,循环结构是核心的一部分,用于控制节点的行为和处理消息。下面是一些常用的循环结构及其用途和用法: while (ros::ok()) 循环: 用途: 保持节点运行,用于执行持续的任务或周期性检查。这个循环确保节点在 ROS 环境下正常运行,直到接…

Java 快速入门

简介 跨平台性&#xff1a;Java 最大的优势之一就是跨平台性&#xff0c;即一份 Java 程序可以在多平台上运行&#xff0c;而无需重写。 简单易学&#xff1a;Java 的语法和面向对象的开发方式非常简单易学。 安全性&#xff1a;Java 对于安全性的处理非常慎重&#xff0c;对…

接口测试和功能测试

本文主要分为两个部分&#xff1a; 第一部分&#xff1a;主要从问题出发&#xff0c;引入接口测试的相关内容并与前端测试进行简单对比&#xff0c;总结两者之前的区别与联系。但该部分只交代了怎么做和如何做&#xff1f;并没有解释为什么要做&#xff1f; 第二部分&#xf…