我们学习了 Nest 如何自定义 logger,也学习了 Winston 的使用。
那如何在 Nest 里集成 Winston 呢?
这节我们来实现下。
nest new nest-winston-test
创建个 nest 项目。
在 src 添加一个 MyLogger.ts
import { LoggerService, LogLevel } from '@nestjs/common';
export class MyLogger implements LoggerService {
log(message: string, context: string) {
console.log(`---log---[${context}]---`, message)
}
error(message: string, context: string) {
console.log(`---error---[${context}]---`, message)
}
warn(message: string, context: string) {
console.log(`---warn---[${context}]---`, message)
}
}
然后在 main.ts 里引入:
app.useLogger(new MyLogger());
把服务跑起来:
npm run start:dev
现在的 logger 就换成我们自己的了。
然后在 AppController 里添加 logger:
浏览器访问下:
这样就完成了 logger 的自定义。
接下来只要换成 winston 的 logger 就好了。
安装 winston:
npm install --save winston
然后改下 MyLogger:
import { ConsoleLogger, LoggerService, LogLevel } from '@nestjs/common';
import { createLogger, format, Logger, transports } from 'winston';
export class MyLogger implements LoggerService {
private logger: Logger;
constructor() {
this.logger = createLogger({
level: 'debug',
format: format.combine(
format.colorize(),
format.simple()
),
transports: [
new transports.Console()
]
});
}
log(message: string, context: string) {
this.logger.log('info', `[${context}] ${message}`);
}
error(message: string, context: string) {
this.logger.log('error', `[${context}] ${message}`);
}
warn(message: string, context: string) {
this.logger.log('warn', `[${context}] ${message}`);
}
}
把 console.log 换成 winston 的 logger。
再跑下:
现在的日志就是 winston 的了。
只不过和 nest 原本的日志格式不大一样。
这个简单,我们自己写一下这种格式就好了。
安装 dayjs 格式化日期:
npm install --save dayjs
安装 chalk 来打印颜色:
npm install --save chalk@4
注意:这里用的是 chalk 4.x 的版本。
然后来实现下 nest 日志的格式:
import { ConsoleLogger, LoggerService, LogLevel } from '@nestjs/common';
import * as chalk from 'chalk';
import * as dayjs from 'dayjs';
import { createLogger, format, Logger, transports } from 'winston';
export class MyLogger implements LoggerService {
private logger: Logger;
constructor() {
super();
this.logger = createLogger({
level: 'debug',
transports: [
new transports.Console({
format: format.combine(
format.colorize(),
format.printf(({context, level, message, time}) => {
const appStr = chalk.green(`[NEST]`);
const contextStr = chalk.yellow(`[${context}]`);
return `${appStr} ${time} ${level} ${contextStr} ${message} `;
})
),
})
]
});
}
log(message: string, context: string) {
const time = dayjs(Date.now()).format('YYYY-MM-DD HH:mm:ss');
this.logger.log('info', message, { context, time });
}
error(message: string, context: string) {
const time = dayjs(Date.now()).format('YYYY-MM-DD HH:mm:ss');
this.logger.log('info', message, { context, time });
}
warn(message: string, context: string) {
const time = dayjs(Date.now()).format('YYYY-MM-DD HH:mm:ss');
this.logger.log('info', message, { context, time });
}
}
这里用到了 printf 的 format 函数,它可以自定义打印的日志格式。
我们用 chalk 加上了颜色,并且打印了 dayjs 格式化的时间。
效果是这样的:
是不是和 nest 原本的日志很像了?
然后我们再加一个 File 的 transport。
指定为 json 格式,加上时间戳:
new transports.File({
format: format.combine(
format.timestamp(),
format.json()
),
filename: '111.log',
dirname: 'log'
})
console 的日志是这样的:
file 的日志是这样的:
这样,我们就完成了 nest 和 winston 的集成。
我们还可以进一步把它封装成一个动态模块。
import { DynamicModule, Global, Module } from '@nestjs/common';
import { LoggerOptions, createLogger } from 'winston';
import { MyLogger } from './MyLogger';
export const WINSTON_LOGGER_TOKEN = 'WINSTON_LOGGER';
@Global()
@Module({})
export class WinstonModule {
public static forRoot(options: LoggerOptions): DynamicModule {
return {
module: WinstonModule,
providers: [
{
provide: WINSTON_LOGGER_TOKEN,
useValue: new MyLogger(options)
}
],
exports: [
WINSTON_LOGGER_TOKEN
]
};
}
}
添加 forRoot 方法,接收 winston 的 createLogger 方法的参数,返回动态模块的 providers、exports。
用 useValue 创建 logger 对象作为 provider。
这里的 MyLogger 是之前那个复制过来的,但需要改一下 constructor:
constructor(options) {
this.logger = createLogger(options)
}
然后在 AppModule 引入下:
import { Module } from '@nestjs/common';
import { AppController } from './app.controller';
import { AppService } from './app.service';
import { WinstonModule } from './winston/winston.module';
import { transports, format } from 'winston';
import * as chalk from 'chalk';
@Module({
imports: [WinstonModule.forRoot({
level: 'debug',
transports: [
new transports.Console({
format: format.combine(
format.colorize(),
format.printf(({context, level, message, time}) => {
const appStr = chalk.green(`[NEST]`);
const contextStr = chalk.yellow(`[${context}]`);
return `${appStr} ${time} ${level} ${contextStr} ${message} `;
})
),
}),
new transports.File({
format: format.combine(
format.timestamp(),
format.json()
),
filename: '111.log',
dirname: 'log'
})
]
})],
controllers: [AppController],
providers: [AppService],
})
export class AppModule {}
之后改一下 main.ts 里用的 logger:
app.useLogger(app.get(WINSTON_LOGGER_TOKEN));
功能正常:
只不过现在就没必要每次都 new 了:
改成 inject 的方式,始终使用同一个实例,性能更好:
@Inject(WINSTON_LOGGER_TOKEN)
private logger;
当然,其实这个模块没必要自己封装,社区有已经封装好的可以直接用:nest-winston
36w 的周下载量:
用法也和我们实现的差不多。
案例代码在小册仓库。
总结
这节我们集成了 nest 和 winston。
前面我们学过了如何自定义 Nest 的 logger,现在只要在 Logger 的实现里改成 winston 的 logger 就好了。
只是想要保持 nest 原本日志的格式,需要用 printf 自定义。我们使用 dayjs + chalk 自定义了 winston 的日志格式。
当然,打印到 File 的日志,依然是 json 的。
之后封装了个动态模块,在 forRoot 方法里传入 options,模块内创建 winston 的 logger 实例。并且这个模块声明为全局模块。
这样,在应用的各处都可以注入我们自定义的基于 winston 的 logger 了。
不过项目里没必要自己写,用 nest-winston 就好了。