前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
工具
TVP
发布
社区首页 >专栏 >基于Egg框架的日志链路追踪实践

基于Egg框架的日志链路追踪实践

作者头像
五月君
发布2019-07-12 16:26:12
1.9K0
发布2019-07-12 16:26:12
举报
文章被收录于专栏:Nodejs技术栈Nodejs技术栈

作者简介:五月君,Nodejs Developer,热爱技术、喜欢分享的 90 后青年,公众号 “Nodejs技术栈”,Github 开源项目 https://www.nodejs.red

快速导航

  • [Logger-Custom] 需求背景
  • [Logger-Custom] 自定义日志插件开发
  • [Logger-Custom] 项目扩展
  • [Logger-Custom] 项目应用
  • [Logrotator] 日志切割

需求背景

实现全链路日志追踪,便于日志监控、问题排查、接口响应耗时数据统计等,首先 API 接口服务接收到调用方请求,根据调用方传的 traceId,在该次调用链中处理业务时,如需打印日志的,日志信息按照约定的规范进行打印,并记录 traceId,实现日志链路追踪。

  • 日志路径约定
代码语言:javascript
复制
/var/logs/${projectName}/bizLog/${projectName}-yyyyMMdd.log
  • 日志格式约定
代码语言:javascript
复制
日志时间[]traceId[]服务端IP[]客户端IP[]日志级别[]日志内容

采用 Egg.js 框架 egg-logger 中间件,在实现过程中发现对于按照以上日志格式打印是无法满足需求的(至少目前我还没找到可实现方式),如果要自己实现,可能要自己造轮子了,好在官方的 egg-logger 中间件提供了自定义日志扩展功能,参考 高级自定义日志:https://eggjs.org/zh-cn/core/logger.html,本身也提供了日志分割、多进程日志处理等功能。

egg-logger 提供了多种传输通道,我们的需求主要是对请求的业务日志自定义格式存储,主要用到 fileTransport 和 consoleTransport 两个通道,分别打印日志到文件和终端。

自定义日志插件开发

基于 egg-logger 定制开发一个插件项目,参考 插件开发:https://eggjs.org/zh-cn/advanced/plugin.html,以下以 egg-logger-custom 为项目,展示核心代码编写

  • 编写logger.js

egg-logger-custom/lib/logger.js

代码语言:javascript
复制
const moment = require('moment');const FileTransport = require('egg-logger').FileTransport;const utils = require('./utils');const util = require('util');
/** * 继承 FileTransport */class AppTransport extends FileTransport {    constructor(options, ctx) {        super(options);
        this.ctx = ctx; // 得到每次请求的上下文    }
    log(level, args, meta) {        // 获取自定义格式消息        const customMsg = this.messageFormat({            level,        });
        // 针对 Error 消息打印出错误的堆栈        if (args[0] instanceof Error) {            const err = args[0] || {};            args[0] = util.format('%s: %s\n%s\npid: %s\n', err.name, err.message, err.stack, process.pid);        } else {            args[0] = util.format(customMsg, args[0]);        }
        // 这个是必须的,否则日志文件不会写入        super.log(level, args, meta);    }
    /**     * 自定义消息格式     * 可以根据自己的业务需求自行定义     * @param { String } level     */    messageFormat({        level    }) {        const { ctx } = this;        const params = JSON.stringify(Object.assign({}, ctx.request.query, ctx.body));
        return [            moment().format('YYYY/MM/DD HH:mm:ss'),            ctx.request.get('traceId'),            utils.serviceIPAddress,            utils.clientIPAddress(ctx.req),            level,        ].join(utils.loggerDelimiter) + utils.loggerDelimiter;    }}
module.exports = AppTransport;
  • 工具

egg-logger-custom/lib/utils.js

代码语言:javascript
复制
const interfaces = require('os').networkInterfaces();
module.exports = {
    /**     * 日志分隔符     */    loggerDelimiter: '[]',
    /**     * 获取当前服务器IP     */    serviceIPAddress: (() => {        for (const devName in interfaces) {            const iface = interfaces[devName];
            for (let i = 0; i < iface.length; i++) {                const alias = iface[i];
                if (alias.family === 'IPv4' && alias.address !== '127.0.0.1' && !alias.internal) {                    return alias.address;                }            }        }    })(),
    /**     * 获取当前请求客户端IP     */    clientIPAddress: req => {        const address = req.headers['x-forwarded-for'] || // 判断是否有反向代理 IP        req.connection.remoteAddress || // 判断 connection 的远程 IP        req.socket.remoteAddress || // 判断后端的 socket 的 IP        req.connection.socket.remoteAddress;
        return address.replace(/::ffff:/ig, '');    }}
  • 初始化 Logger

egg-logger-custom/app.js

代码语言:javascript
复制
const Logger = require('egg-logger').Logger;const ConsoleTransport = require('egg-logger').ConsoleTransport;const AppTransport = require('./app/logger');
module.exports = (ctx, options) => {    const logger = new Logger();
    logger.set('file', new AppTransport({        level: options.fileLoggerLevel || 'INFO',        file: `/var/logs/${options.appName}/bizLog/${options.appName}.log`,    }, ctx));
    logger.set('console', new ConsoleTransport({        level: options.consoleLevel || 'INFO',    }));
    return logger;}

以上对于日志定制格式开发已经好了,如果你有实际业务需要可以根据自己团队的需求,封装为团队内部的一个 npm 中间件来使用。

项目扩展

自定义日志中间件封装好之后,在实际项目应用中我们还需要一步操作,Egg 提供了 框架扩展 功能,包含五项:Application、Context、Request、Response、Helper,可以对这几项进行自定义扩展,对于日志因为每次日志记录我们需要记录当前请求携带的 traceId 做一个链路追踪,需要用到 Context(是 Koa 的请求上下文) 扩展项。

新建 app/extend/context.js 文件

代码语言:javascript
复制
const AppLogger = require('egg-logger-custom'); // 上面定义的中间件
module.exports = {    get logger() { // 名字自定义 也可以是 customLogger        return AppLogger(this, {            appName: 'test', // 项目名称            consoleLevel: 'DEBUG', // 终端日志级别            fileLoggerLevel: 'DEBUG', // 文件日志级别        });    }}

建议:对于日志级别,可以采用配置中心如 Consul 进行配置,上线时日志级别设置为 INFO,当需要生产问题排查时,可以动态开启 DEBUG 模式。关于 Consul 可以关注我之前写的 服务注册发现 Consul 系列

项目应用

错误日志记录,直接会将错误日志完整堆栈信息记录下来,并且输出到 errorLog 中,为了保证异常可追踪,必须保证所有抛出的异常都是 Error 类型,因为只有 Error 类型才会带上堆栈信息,定位到问题。

代码语言:javascript
复制
const Controller = require('egg').Controller;
class ExampleController extends Controller {    async list() {        const { ctx } = this;
        ctx.logger.error(new Error('程序异常!'));
        ctx.logger.debug('测试');
        ctx.logger.info('测试');    }}

最终日志打印格式如下所示:

代码语言:javascript
复制
2019/05/30 01:50:21[]d373c38a-344b-4b36-b931-1e8981aef14f[]192.168.1.20[]221.69.245.153[]INFO[]测试

日志切割

框架提供了 egg-logrotator 中间件,默认切割为按天切割,其它方式可参考官网自行配置。

  • 框架默认日志路径

egg-logger 模块 lib/egg/config/config.default.js

代码语言:javascript
复制
config.logger = {    dir: path.join(appInfo.root, 'logs', appInfo.name),    ...};
  • 自定义日志目录

很简单按照我们的需求在项目配置文件重新定义 logger 的 dir 路径

代码语言:javascript
复制
config.logger = {    dir: /var/logs/test/bizLog/}

这样是否就可以呢?按照我们上面自定义的日志文件名格式( ${projectName}-yyyyMMdd.log),貌似是不行的,在日志分割过程中默认的文件名格式为 .log.YYYY-MM-DD,参考源码

https://github.com/eggjs/egg-logrotator/blob/master/app/lib/day_rotator.js

代码语言:javascript
复制
 _setFile(srcPath, files) {    // don't rotate logPath in filesRotateBySize    if (this.filesRotateBySize.indexOf(srcPath) > -1) {      return;    }
    // don't rotate logPath in filesRotateByHour    if (this.filesRotateByHour.indexOf(srcPath) > -1) {      return;    }
    if (!files.has(srcPath)) {      // allow 2 minutes deviation      const targetPath = srcPath + moment()        .subtract(23, 'hours')        .subtract(58, 'minutes')        .format('.YYYY-MM-DD'); // 日志格式定义      debug('set file %s => %s', srcPath, targetPath);      files.set(srcPath, { srcPath, targetPath });    } }
  • 日志分割扩展

中间件 egg-logrotator 预留了扩展接口,对于自定义的日志文件名,可以用框架提供的 app.LogRotator 做一个定制。

app/schedule/custom.js

代码语言:javascript
复制
const moment = require('moment');
module.exports = app => {    const rotator = getRotator(app);
    return {        schedule: {            type: 'worker', // only one worker run this task            cron: '1 0 0 * * *', // run every day at 00:00        },        async task() {            await rotator.rotate();        }    };};
function getRotator(app) {    class CustomRotator extends app.LogRotator {        async getRotateFiles() {            const files = new Map();            const srcPath = `/var/logs/test/bizLog/test.log`;            const targetPath = `/var/logs/test/bizLog/test-${moment().subtract(1, 'days').format('YYYY-MM-DD')}.log`;            files.set(srcPath, { srcPath, targetPath });            return files;        }    }
    return new CustomRotator({ app });}

经过分割之后文件展示如下:

代码语言:javascript
复制
$ ls -lh /var/logs/test/bizLog/total 188K-rw-r--r-- 1 root root 135K Jun  1 11:00 test-2019-06-01.log-rw-r--r-- 1 root root  912 Jun  2 09:44 test-2019-06-02.log-rw-r--r-- 1 root root  40K Jun  3 11:49 test.log

扩展:基于以上日志格式,可以采用 ELK 做日志搜集、分析、检索。

作者推荐

  • 个人博客:https://www.nodejs.red
  • 慕课:https://imooc.com/article/287529
  • 公众号:Nodejs技术栈
本文参与 腾讯云自媒体分享计划,分享自微信公众号。
原始发表:2019-06-03,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 Nodejs技术栈 微信公众号,前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体分享计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
0 条评论
热度
最新
推荐阅读
目录
  • 快速导航
  • 需求背景
  • 自定义日志插件开发
  • 项目扩展
  • 项目应用
  • 日志切割
相关产品与服务
日志服务
日志服务(Cloud Log Service,CLS)是腾讯云提供的一站式日志服务平台,提供了从日志采集、日志存储到日志检索,图表分析、监控告警、日志投递等多项服务,协助用户通过日志来解决业务运维、服务监控、日志审计等场景问题。
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档