koa-logger
Version:
Logging middleware for koa
158 lines (133 loc) • 3.53 kB
JavaScript
/**
* Module dependencies.
*/
'use strict'
const Counter = require('passthrough-counter')
const humanize = require('humanize-number')
const bytes = require('bytes')
const chalk = require('chalk')
const util = require('util')
/**
* Expose logger.
*/
module.exports = dev
/**
* Color map.
*/
const colorCodes = {
7: 'magenta',
5: 'red',
4: 'yellow',
3: 'cyan',
2: 'green',
1: 'green',
0: 'yellow'
}
/**
* Development logger.
*/
function dev (opts) {
// print to console helper.
const print = (function () {
let transporter
if (typeof opts === 'function') {
transporter = opts
} else if (opts && opts.transporter) {
transporter = opts.transporter
}
return function printFunc (...args) {
const str = util.format(...args)
if (transporter) {
transporter(str, args)
} else {
console.log(...args)
}
}
}())
return async function logger (ctx, next) {
// request
const start = ctx[Symbol.for('request-received.startTime')] ? ctx[Symbol.for('request-received.startTime')].getTime() : Date.now()
print(' ' + chalk.gray('<--') +
' ' + chalk.bold('%s') +
' ' + chalk.gray('%s'),
ctx.method,
ctx.originalUrl)
try {
await next()
} catch (err) {
// log uncaught downstream errors
log(print, ctx, start, null, err)
throw err
}
// calculate the length of a streaming response
// by intercepting the stream with a counter.
// only necessary if a content-length header is currently not set.
const length = ctx.response.length
const body = ctx.body
let counter
if (length == null && body && body.readable) {
ctx.body = body
.pipe(counter = Counter())
.on('error', ctx.onerror)
}
// log when the response is finished or closed,
// whichever happens first.
const res = ctx.res
const onfinish = done.bind(null, 'finish')
const onclose = done.bind(null, 'close')
res.once('finish', onfinish)
res.once('close', onclose)
function done (event) {
res.removeListener('finish', onfinish)
res.removeListener('close', onclose)
log(print, ctx, start, counter ? counter.length : length, null, event)
}
}
}
/**
* Log helper.
*/
function log (print, ctx, start, len, err, event) {
// get the status code of the response
const status = err
? (err.isBoom ? err.output.statusCode : err.status || 500)
: (ctx.status || 404)
// set the color of the status code;
const s = status / 100 | 0
// eslint-disable-next-line
const color = colorCodes.hasOwnProperty(s) ? colorCodes[s] : colorCodes[0]
// get the human readable response length
let length
if (~[204, 205, 304].indexOf(status)) {
length = ''
} else if (len == null) {
length = '-'
} else {
length = bytes(len).toLowerCase()
}
const upstream = err ? chalk.red('xxx')
: event === 'close' ? chalk.yellow('-x-')
: chalk.gray('-->')
print(' ' + upstream +
' ' + chalk.bold('%s') +
' ' + chalk.gray('%s') +
' ' + chalk[color]('%s') +
' ' + chalk.gray('%s') +
' ' + chalk.gray('%s'),
ctx.method,
ctx.originalUrl,
status,
time(start),
length)
}
/**
* Show the response time in a human readable format.
* In milliseconds if less than 10 seconds,
* in seconds otherwise.
*/
function time (start) {
const delta = Date.now() - start
return humanize(delta < 10000
? delta + 'ms'
: Math.round(delta / 1000) + 's')
}