一、新建一个log4js.js配置文件

let path = require('path');

// 日志根目录
let baseLogPath = path.resolve(__dirname, '../logs'); // 错误日志目录
let errorPath = '/error';
// 错误日志文件名
let errorFileName = 'error';
// 错误日志输出完整路径
let errorLogPath = baseLogPath + errorPath + '/' + errorFileName; // 请求日志目录
let reqPath = '/request';
// 请求日志文件名
let reqFileName = 'request';
// 请求日志输出完整路径
let reqLogPath = baseLogPath + reqPath + '/' + reqFileName; // 响应日志目录
let responsePath = '/response';
// 响应日志文件名
let responseFileName = 'response';
// 响应日志输出完整路径
let responseLogPath = baseLogPath + responsePath + '/' + responseFileName; module.exports = {
// 日志格式等设置
appenders:
{
console: {
type: 'console' },
errorLogger: {
type: 'dateFile',
filename: errorLogPath,
pattern: '-yyyy-MM-dd-hh.log',
alwaysIncludePattern: true,
encoding: 'utf-8',
maxLogSize: 1000,
numBackups: 3,
path: errorPath,
layout: {
type: 'basic'
}
},
http: {
type: 'dateFile',
filename: reqLogPath,
pattern: '-yyyy-MM-dd-hh.log',
alwaysIncludePattern: true,
encoding: 'utf-8',
maxLogSize: 1000,
numBackups: 3,
path: reqPath,
layout: {
type: 'basic'// 'messagePassThrough'
}
},
resLogger: {
type: 'dateFile',
filename: responseLogPath,
pattern: '-yyyy-MM-dd-hh.log',
alwaysIncludePattern: true,
encoding: 'utf-8',
maxLogSize: 1000,
numBackups: 3,
path: responsePath,
layout: {
type: 'basic'
}
}
},
// 供外部调用的名称和对应设置定义
categories: {
default: {
appenders: ['console'], level: 'all'
},
resLogger: {
appenders: ['resLogger'], level: 'info'
},
errorLogger: {
appenders: ['errorLogger'], level: 'error'
},
http: {
appenders: ['http'], level: 'info'
}
},
baseLogPath,
replaceConsole: true
};

二、新建log4Util.js

var log4js = require('log4js');

var logConfig = require('../config/log4js');

// 加载配置文件
log4js.configure(logConfig); var logUtil = {};
// 调用预先定义的日志名称
var resLogger = log4js.getLogger('resLogger');
var reqLogger = log4js.getLogger('http');
var errorLogger = log4js.getLogger('errorLogger');
var consoleLogger = log4js.getLogger(); // 封装错误日志
logUtil.logError = function (ctx, error, resTime) {
if (ctx && error) {
errorLogger.error(formatError(ctx, error, resTime));
}
}; // 封装请求日志
logUtil.reqLog = function (ctx, resTime) {
if (ctx) {
reqLogger.info(formatReqLog(ctx, resTime));
}
};
// 封装响应日志
logUtil.logResponse = function (ctx, resTime) {
if (ctx) {
resLogger.info(formatRes(ctx, resTime));
}
}; logUtil.logInfo = function (info) {
if (info) {
consoleLogger.info(formatInfo(info));
}
}; var formatInfo = function (info) {
var logText = '';
// 响应日志开始
logText += '\n' + '***************info log start ***************' + '\n'; // 响应内容
logText += 'info detail: ' + '\n' + JSON.stringify(info) + '\n'; // 响应日志结束
logText += '*************** info log end ***************' + '\n'; return logText;
}; // 格式化响应日志
var formatRes = function (ctx, resTime) {
var logText = '';
// 响应日志开始
logText += '\n' + '*************** response log start ***************' + '\n'; // 添加请求日志
logText += formatReqLog(ctx.request, resTime); // 响应状态码
logText += 'response status: ' + ctx.status + '\n'; // 响应内容
logText += 'response body: ' + '\n' + JSON.stringify(ctx.body) + '\n'; // 响应日志结束
logText += '*************** response log end ***************' + '\n'; return logText;
}; // 格式化错误日志
var formatError = function (ctx, err, resTime) {
var logText = ''; // 错误信息开始
logText += '\n' + '*************** error log start ***************' + '\n'; // 添加请求日志
logText += formatReqLog(ctx.request, resTime); // 错误名称
logText += 'err name: ' + err.name + '\n';
// 错误信息
logText += 'err message: ' + err.message + '\n';
// 错误详情
logText += 'err stack: ' + err.stack + '\n'; // 错误信息结束
logText += '*************** error log end ***************' + '\n'; return logText;
}; // 格式化请求日志
var formatReqLog = function (req, resTime) {
var logText = ''; var method = req.method;
// 访问方法
logText += '\n' + 'request method: ' + method + '\n'; // 请求原始地址
logText += 'request originalUrl: ' + req.originalUrl + '\n'; // 客户端ip
logText += 'request client ip: ' + req.ip + '\n'; // 开始时间
// var startTime;
// 请求参数
if (method === 'GET') {
logText += 'request query: ' + JSON.stringify(req.query) + '\n';
// startTime = req.query.requestStartTime;
} else {
logText += 'request body: ' + '\n' + JSON.stringify(req.body) + '\n';
// startTime = req.body.requestStartTime;
}
// 服务器响应时间
logText += 'response time: ' + resTime + '\n'; return logText;
}; module.exports = () => {
return async (ctx, next) => {
ctx.logger = logUtil;
ctx.logger.reqLog(ctx, 0);
await next();
};
};

 三、再app.js引入log4Util.js

// logger
app.use(async (ctx, next) => {
const start = new Date();
let ms = new Date() - start;
await next();
try {
// 开始进入到下一个中间件
if (ctx.status === 404) {
ctx.throw(404);
}
ms = new Date() - start;
// 记录响应日志
ctx.logger.logResponse(ctx, ms);
} catch (error) {
ms = new Date() - start;
// 记录异常日志
ctx.logger.logError(ctx, error, ms);
}
}); // routes
app.use(index.routes());
app.use(index.allowedMethods());

  

koa中 log4js使用的更多相关文章

  1. Koa与Node.js开发实战(3)——Nunjucks模板在Koa中的应用(视频演示)

    技术架构: ​ 在Koa中应用Nunjucks,需要先把Nunjucks集成为符合Koa规格的中间件(Middleware),从本质上来讲,集成后的中间件的作用是给上下文对象绑定一个render(vi ...

  2. Koa 中 ejs 模板的使用

    ejs的基本使用 安装 koa-views 和 ejs npm install --save koa-views/cnpm install --save koa-views npm install e ...

  3. 从前端中的IOC理念理解koa中的app.use()

    忙里偷闲,打开平时关注的前端相关的网站,浏览最近最新的前端动态.佼佼者,平凡的我做不到,但还是要争取不做落后者. 前端中的IoC理念,看到这个标题就被吸引了.IoC 理念,不认识呢,点击去一看,果然没 ...

  4. koa中返回404并且刷新后才正常的解决方案

    概述 这几天学习koa2,有一些心得,记录下来,供以后开发时参考,相信对其他人也有用. 起因 这几天学习koa2,写的代码执行时有一个奇怪的bug:明明能够返回数据,却有时正常返回数据,有时偏偏给你返 ...

  5. Koa 中实现 chunked 数据传输

    有关于 Transfer-Encoding:chunked 类型的响应,参见之前的文章HTTP 响应的分块传输.这里看 Koa 中如何实现. Koa 中请求返回的处理 虽然官方文档有描述说明不建议直接 ...

  6. koa 基础(十六)koa 中 session 的使用

    1.app.js /** * koa 中 session 的使用 * 1.npm install koa-session --save * 2.const session = require('koa ...

  7. koa 基础(十一)koa 中 koa-bodyparser 中间件获取表单提交的数据

    1.app.js /** * koa 中 koa-bodyparser 中间件获取表单提交的数据 * 1.npm install --save koa-bodyparser * 2.引入 const ...

  8. koa 基础(十)原生node.js 在 koa 中获取表单提交的数据

    1.app.js // 引入模块 const Koa = require('koa'); const router = require('koa-router')(); /*引入是实例化路由 推荐*/ ...

  9. node.js中log4js的使用

    以前用过forever进程守护的日志记录到指定文件,但是只能保存到一个文件中不能分片,这样到只日志文件越来越大, forever start -s -l ./forever.log app.js -l ...

随机推荐

  1. python3使用ConfigParser从配置文件中获取列表

    使用python3使用ConfigParser从配置文件中获取列表 testConfig.py #!/usr/bin/env python # coding=utf- __author__ = 'St ...

  2. centos7防火墙相关

    selinux(保护文件安全) 安全增强型 Linux(Security-Enhanced Linux)简称 SELinux,它是一个 Linux 内核模块,也是 Linux 的一个安全子系统. SE ...

  3. 复杂sql书写方法

    给你一个复杂sql连接不同的表,多个嵌套查询条件等的语句时,你是非常的胆怯由于对语法的不熟悉以及没有经验和自信,现在我们来学习一下如何写复杂的sql,我们把它分解为很多小的步骤进行 一.集中最后的输出 ...

  4. (三)AppScan扫描策略的选择

    使用 AppScan 进行扫描 针对大型网站的扫描,我们按照戴明环 PDCA 的方法论来进行规划和讨论,建议 AppScan 使用步骤:计划(Plan).执行(Do).检查(check).分析(Ana ...

  5. 拒绝流量劫持,全面使用 HTTPS!

    最近收到数个 BootCDN 用户的反馈:某些地区的宽带运营商劫持了部分 BootCDN 上的文件,并篡改文件加入了广告代码. 这种方式的流量劫持属于中间人攻击(Man-in-the-Middle A ...

  6. python 传参

    python不允许程序员选择采用传值还是传引用.Python参数传递采用的肯定是“传对象引用”的方式.这种方式相当于传值和传引用的一种综合.如果函数收到的是一个可变对象(比如字典或者列表)的引用,就能 ...

  7. Java8-Executors-No.02

    import java.util.concurrent.ExecutionException; import java.util.concurrent.ExecutorService; import ...

  8. stl常数测试

    如图: 数组的常数约为9.

  9. Vue项目中的文件/文件夹命名规范

    Vue项目中的文件/文件夹命名规范 0.2262018.09.21 16:01:09字数 820阅读 6979 文件或文件夹的命名遵循以下原则: index.js 或者 index.vue,统一使用小 ...

  10. 如何实现对ArrayList排序 sort()

    package com.collection; import java.util.ArrayList;import java.util.Collections;import java.util.Com ...