554 lines
13 KiB
JavaScript
554 lines
13 KiB
JavaScript
|
'use strict'
|
||
|
const test = require('tape')
|
||
|
const fresh = require('import-fresh')
|
||
|
const pinoStdSerializers = require('pino-std-serializers')
|
||
|
const pino = require('../browser')
|
||
|
|
||
|
levelTest('fatal')
|
||
|
levelTest('error')
|
||
|
levelTest('warn')
|
||
|
levelTest('info')
|
||
|
levelTest('debug')
|
||
|
levelTest('trace')
|
||
|
|
||
|
test('silent level', ({ end, fail, pass }) => {
|
||
|
const instance = pino({
|
||
|
level: 'silent',
|
||
|
browser: { write: fail }
|
||
|
})
|
||
|
instance.info('test')
|
||
|
const child = instance.child({ test: 'test' })
|
||
|
child.info('msg-test')
|
||
|
// use setTimeout because setImmediate isn't supported in most browsers
|
||
|
setTimeout(() => {
|
||
|
pass()
|
||
|
end()
|
||
|
}, 0)
|
||
|
})
|
||
|
|
||
|
test('enabled false', ({ end, fail, pass }) => {
|
||
|
const instance = pino({
|
||
|
enabled: false,
|
||
|
browser: { write: fail }
|
||
|
})
|
||
|
instance.info('test')
|
||
|
const child = instance.child({ test: 'test' })
|
||
|
child.info('msg-test')
|
||
|
// use setTimeout because setImmediate isn't supported in most browsers
|
||
|
setTimeout(() => {
|
||
|
pass()
|
||
|
end()
|
||
|
}, 0)
|
||
|
})
|
||
|
|
||
|
test('throw if creating child without bindings', ({ end, throws }) => {
|
||
|
const instance = pino()
|
||
|
throws(() => instance.child())
|
||
|
end()
|
||
|
})
|
||
|
|
||
|
test('stubs write, flush and ee methods on instance', ({ end, ok, is }) => {
|
||
|
const instance = pino()
|
||
|
|
||
|
ok(isFunc(instance.setMaxListeners))
|
||
|
ok(isFunc(instance.getMaxListeners))
|
||
|
ok(isFunc(instance.emit))
|
||
|
ok(isFunc(instance.addListener))
|
||
|
ok(isFunc(instance.on))
|
||
|
ok(isFunc(instance.prependListener))
|
||
|
ok(isFunc(instance.once))
|
||
|
ok(isFunc(instance.prependOnceListener))
|
||
|
ok(isFunc(instance.removeListener))
|
||
|
ok(isFunc(instance.removeAllListeners))
|
||
|
ok(isFunc(instance.listeners))
|
||
|
ok(isFunc(instance.listenerCount))
|
||
|
ok(isFunc(instance.eventNames))
|
||
|
ok(isFunc(instance.write))
|
||
|
ok(isFunc(instance.flush))
|
||
|
|
||
|
is(instance.on(), undefined)
|
||
|
|
||
|
end()
|
||
|
})
|
||
|
|
||
|
test('exposes levels object', ({ end, same }) => {
|
||
|
same(pino.levels, {
|
||
|
values: {
|
||
|
fatal: 60,
|
||
|
error: 50,
|
||
|
warn: 40,
|
||
|
info: 30,
|
||
|
debug: 20,
|
||
|
trace: 10
|
||
|
},
|
||
|
labels: {
|
||
|
10: 'trace',
|
||
|
20: 'debug',
|
||
|
30: 'info',
|
||
|
40: 'warn',
|
||
|
50: 'error',
|
||
|
60: 'fatal'
|
||
|
}
|
||
|
})
|
||
|
|
||
|
end()
|
||
|
})
|
||
|
|
||
|
test('exposes LOG_VERSION', ({ end, is }) => {
|
||
|
is(pino.LOG_VERSION, 1)
|
||
|
|
||
|
end()
|
||
|
})
|
||
|
|
||
|
test('exposes faux stdSerializers', ({ end, ok, same }) => {
|
||
|
ok(pino.stdSerializers)
|
||
|
// make sure faux stdSerializers match pino-std-serializers
|
||
|
for (const serializer in pinoStdSerializers) {
|
||
|
ok(pino.stdSerializers[serializer], `pino.stdSerializers.${serializer}`)
|
||
|
}
|
||
|
// confirm faux methods return empty objects
|
||
|
same(pino.stdSerializers.req(), {})
|
||
|
same(pino.stdSerializers.mapHttpRequest(), {})
|
||
|
same(pino.stdSerializers.mapHttpResponse(), {})
|
||
|
same(pino.stdSerializers.res(), {})
|
||
|
// confirm wrapping function is a passthrough
|
||
|
const noChange = { foo: 'bar', fuz: 42 }
|
||
|
same(pino.stdSerializers.wrapRequestSerializer(noChange), noChange)
|
||
|
same(pino.stdSerializers.wrapResponseSerializer(noChange), noChange)
|
||
|
end()
|
||
|
})
|
||
|
|
||
|
test('exposes err stdSerializer', ({ end, ok }) => {
|
||
|
ok(pino.stdSerializers.err)
|
||
|
ok(pino.stdSerializers.err(Error()))
|
||
|
end()
|
||
|
})
|
||
|
|
||
|
consoleMethodTest('error')
|
||
|
consoleMethodTest('fatal', 'error')
|
||
|
consoleMethodTest('warn')
|
||
|
consoleMethodTest('info')
|
||
|
consoleMethodTest('debug')
|
||
|
consoleMethodTest('trace')
|
||
|
absentConsoleMethodTest('error', 'log')
|
||
|
absentConsoleMethodTest('warn', 'error')
|
||
|
absentConsoleMethodTest('info', 'log')
|
||
|
absentConsoleMethodTest('debug', 'log')
|
||
|
absentConsoleMethodTest('trace', 'log')
|
||
|
|
||
|
// do not run this with airtap
|
||
|
if (process.title !== 'browser') {
|
||
|
test('in absence of console, log methods become noops', ({ end, ok }) => {
|
||
|
var console = global.console
|
||
|
delete global.console
|
||
|
const instance = fresh('../browser')()
|
||
|
global.console = console
|
||
|
ok(fnName(instance.log).match(/noop/))
|
||
|
ok(fnName(instance.fatal).match(/noop/))
|
||
|
ok(fnName(instance.error).match(/noop/))
|
||
|
ok(fnName(instance.warn).match(/noop/))
|
||
|
ok(fnName(instance.info).match(/noop/))
|
||
|
ok(fnName(instance.debug).match(/noop/))
|
||
|
ok(fnName(instance.trace).match(/noop/))
|
||
|
end()
|
||
|
})
|
||
|
}
|
||
|
|
||
|
test('opts.browser.asObject logs pino-like object to console', ({ end, ok, is }) => {
|
||
|
var info = console.info
|
||
|
console.info = function (o) {
|
||
|
is(o.level, 30)
|
||
|
is(o.msg, 'test')
|
||
|
ok(o.time)
|
||
|
console.info = info
|
||
|
}
|
||
|
const instance = require('../browser')({
|
||
|
browser: {
|
||
|
asObject: true
|
||
|
}
|
||
|
})
|
||
|
|
||
|
instance.info('test')
|
||
|
end()
|
||
|
})
|
||
|
|
||
|
test('opts.browser.write func log single string', ({ end, ok, is }) => {
|
||
|
const instance = pino({
|
||
|
browser: {
|
||
|
write: function (o) {
|
||
|
is(o.level, 30)
|
||
|
is(o.msg, 'test')
|
||
|
ok(o.time)
|
||
|
}
|
||
|
}
|
||
|
})
|
||
|
instance.info('test')
|
||
|
|
||
|
end()
|
||
|
})
|
||
|
|
||
|
test('opts.browser.write func string joining', ({ end, ok, is }) => {
|
||
|
const instance = pino({
|
||
|
browser: {
|
||
|
write: function (o) {
|
||
|
is(o.level, 30)
|
||
|
is(o.msg, 'test test2 test3')
|
||
|
ok(o.time)
|
||
|
}
|
||
|
}
|
||
|
})
|
||
|
instance.info('test', 'test2', 'test3')
|
||
|
|
||
|
end()
|
||
|
})
|
||
|
|
||
|
test('opts.browser.write func string joining when asObject is true', ({ end, ok, is }) => {
|
||
|
const instance = pino({
|
||
|
browser: {
|
||
|
asObject: true,
|
||
|
write: function (o) {
|
||
|
is(o.level, 30)
|
||
|
is(o.msg, 'test test2 test3')
|
||
|
ok(o.time)
|
||
|
}
|
||
|
}
|
||
|
})
|
||
|
instance.info('test', 'test2', 'test3')
|
||
|
|
||
|
end()
|
||
|
})
|
||
|
|
||
|
test('opts.browser.write func string joining when asObject is true', ({ end, ok, is }) => {
|
||
|
const instance = pino({
|
||
|
browser: {
|
||
|
asObject: true,
|
||
|
write: function (o) {
|
||
|
is(o.level, 30)
|
||
|
is(o.msg, 'test test2 test3')
|
||
|
ok(o.time)
|
||
|
}
|
||
|
}
|
||
|
})
|
||
|
instance.info('test', 'test2', 'test3')
|
||
|
|
||
|
end()
|
||
|
})
|
||
|
|
||
|
test('opts.browser.write func string object joining', ({ end, ok, is }) => {
|
||
|
const instance = pino({
|
||
|
browser: {
|
||
|
write: function (o) {
|
||
|
is(o.level, 30)
|
||
|
is(o.msg, 'test {"test":"test2"} {"test":"test3"}')
|
||
|
ok(o.time)
|
||
|
}
|
||
|
}
|
||
|
})
|
||
|
instance.info('test', { test: 'test2' }, { test: 'test3' })
|
||
|
|
||
|
end()
|
||
|
})
|
||
|
|
||
|
test('opts.browser.write func string object joining when asObject is true', ({ end, ok, is }) => {
|
||
|
const instance = pino({
|
||
|
browser: {
|
||
|
asObject: true,
|
||
|
write: function (o) {
|
||
|
is(o.level, 30)
|
||
|
is(o.msg, 'test {"test":"test2"} {"test":"test3"}')
|
||
|
ok(o.time)
|
||
|
}
|
||
|
}
|
||
|
})
|
||
|
instance.info('test', { test: 'test2' }, { test: 'test3' })
|
||
|
|
||
|
end()
|
||
|
})
|
||
|
|
||
|
test('opts.browser.write func string interpolation', ({ end, ok, is }) => {
|
||
|
const instance = pino({
|
||
|
browser: {
|
||
|
write: function (o) {
|
||
|
is(o.level, 30)
|
||
|
is(o.msg, 'test2 test ({"test":"test3"})')
|
||
|
ok(o.time)
|
||
|
}
|
||
|
}
|
||
|
})
|
||
|
instance.info('%s test (%j)', 'test2', { test: 'test3' })
|
||
|
|
||
|
end()
|
||
|
})
|
||
|
|
||
|
test('opts.browser.write func number', ({ end, ok, is }) => {
|
||
|
const instance = pino({
|
||
|
browser: {
|
||
|
write: function (o) {
|
||
|
is(o.level, 30)
|
||
|
is(o.msg, 1)
|
||
|
ok(o.time)
|
||
|
}
|
||
|
}
|
||
|
})
|
||
|
instance.info(1)
|
||
|
|
||
|
end()
|
||
|
})
|
||
|
|
||
|
test('opts.browser.write func log single object', ({ end, ok, is }) => {
|
||
|
const instance = pino({
|
||
|
browser: {
|
||
|
write: function (o) {
|
||
|
is(o.level, 30)
|
||
|
is(o.test, 'test')
|
||
|
ok(o.time)
|
||
|
}
|
||
|
}
|
||
|
})
|
||
|
instance.info({ test: 'test' })
|
||
|
|
||
|
end()
|
||
|
})
|
||
|
|
||
|
test('opts.browser.write obj writes to methods corresponding to level', ({ end, ok, is }) => {
|
||
|
const instance = pino({
|
||
|
browser: {
|
||
|
write: {
|
||
|
error: function (o) {
|
||
|
is(o.level, 50)
|
||
|
is(o.test, 'test')
|
||
|
ok(o.time)
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
})
|
||
|
instance.error({ test: 'test' })
|
||
|
|
||
|
end()
|
||
|
})
|
||
|
|
||
|
test('opts.browser.asObject/write supports child loggers', ({ end, ok, is }) => {
|
||
|
const instance = pino({
|
||
|
browser: {
|
||
|
write (o) {
|
||
|
is(o.level, 30)
|
||
|
is(o.test, 'test')
|
||
|
is(o.msg, 'msg-test')
|
||
|
ok(o.time)
|
||
|
}
|
||
|
}
|
||
|
})
|
||
|
const child = instance.child({ test: 'test' })
|
||
|
child.info('msg-test')
|
||
|
|
||
|
end()
|
||
|
})
|
||
|
|
||
|
test('opts.browser.asObject/write supports child child loggers', ({ end, ok, is }) => {
|
||
|
const instance = pino({
|
||
|
browser: {
|
||
|
write (o) {
|
||
|
is(o.level, 30)
|
||
|
is(o.test, 'test')
|
||
|
is(o.foo, 'bar')
|
||
|
is(o.msg, 'msg-test')
|
||
|
ok(o.time)
|
||
|
}
|
||
|
}
|
||
|
})
|
||
|
const child = instance.child({ test: 'test' }).child({ foo: 'bar' })
|
||
|
child.info('msg-test')
|
||
|
|
||
|
end()
|
||
|
})
|
||
|
|
||
|
test('opts.browser.asObject/write supports child child child loggers', ({ end, ok, is }) => {
|
||
|
const instance = pino({
|
||
|
browser: {
|
||
|
write (o) {
|
||
|
is(o.level, 30)
|
||
|
is(o.test, 'test')
|
||
|
is(o.foo, 'bar')
|
||
|
is(o.baz, 'bop')
|
||
|
is(o.msg, 'msg-test')
|
||
|
ok(o.time)
|
||
|
}
|
||
|
}
|
||
|
})
|
||
|
const child = instance.child({ test: 'test' }).child({ foo: 'bar' }).child({ baz: 'bop' })
|
||
|
child.info('msg-test')
|
||
|
|
||
|
end()
|
||
|
})
|
||
|
|
||
|
test('opts.browser.asObject defensively mitigates naughty numbers', ({ end, pass }) => {
|
||
|
const instance = pino({
|
||
|
browser: { asObject: true, write: () => {} }
|
||
|
})
|
||
|
const child = instance.child({ test: 'test' })
|
||
|
child._childLevel = -10
|
||
|
child.info('test')
|
||
|
pass() // if we reached here, there was no infinite loop, so, .. pass.
|
||
|
|
||
|
end()
|
||
|
})
|
||
|
|
||
|
test('opts.browser.write obj falls back to console where a method is not supplied', ({ end, ok, is }) => {
|
||
|
var info = console.info
|
||
|
console.info = (o) => {
|
||
|
is(o.level, 30)
|
||
|
is(o.msg, 'test')
|
||
|
ok(o.time)
|
||
|
console.info = info
|
||
|
}
|
||
|
const instance = require('../browser')({
|
||
|
browser: {
|
||
|
write: {
|
||
|
error (o) {
|
||
|
is(o.level, 50)
|
||
|
is(o.test, 'test')
|
||
|
ok(o.time)
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
})
|
||
|
instance.error({ test: 'test' })
|
||
|
instance.info('test')
|
||
|
|
||
|
end()
|
||
|
})
|
||
|
|
||
|
function levelTest (name) {
|
||
|
test(name + ' logs', ({ end, is }) => {
|
||
|
var msg = 'hello world'
|
||
|
sink(name, (args) => {
|
||
|
is(args[0], msg)
|
||
|
end()
|
||
|
})
|
||
|
pino({ level: name })[name](msg)
|
||
|
})
|
||
|
|
||
|
test('passing objects at level ' + name, ({ end, is }) => {
|
||
|
var msg = { hello: 'world' }
|
||
|
sink(name, (args) => {
|
||
|
is(args[0], msg)
|
||
|
end()
|
||
|
})
|
||
|
pino({ level: name })[name](msg)
|
||
|
})
|
||
|
|
||
|
test('passing an object and a string at level ' + name, ({ end, is }) => {
|
||
|
var a = { hello: 'world' }
|
||
|
var b = 'a string'
|
||
|
sink(name, (args) => {
|
||
|
is(args[0], a)
|
||
|
is(args[1], b)
|
||
|
end()
|
||
|
})
|
||
|
pino({ level: name })[name](a, b)
|
||
|
})
|
||
|
|
||
|
test('formatting logs as ' + name, ({ end, is }) => {
|
||
|
sink(name, (args) => {
|
||
|
is(args[0], 'hello %d')
|
||
|
is(args[1], 42)
|
||
|
end()
|
||
|
})
|
||
|
pino({ level: name })[name]('hello %d', 42)
|
||
|
})
|
||
|
|
||
|
test('passing error at level ' + name, ({ end, is }) => {
|
||
|
var err = new Error('myerror')
|
||
|
sink(name, (args) => {
|
||
|
is(args[0], err)
|
||
|
end()
|
||
|
})
|
||
|
pino({ level: name })[name](err)
|
||
|
})
|
||
|
|
||
|
test('passing error with a serializer at level ' + name, ({ end, is }) => {
|
||
|
// in browser - should have no effect (should not crash)
|
||
|
var err = new Error('myerror')
|
||
|
sink(name, (args) => {
|
||
|
is(args[0].err, err)
|
||
|
end()
|
||
|
})
|
||
|
const instance = pino({
|
||
|
level: name,
|
||
|
serializers: {
|
||
|
err: pino.stdSerializers.err
|
||
|
}
|
||
|
})
|
||
|
instance[name]({ err: err })
|
||
|
})
|
||
|
|
||
|
test('child logger for level ' + name, ({ end, is }) => {
|
||
|
var msg = 'hello world'
|
||
|
var parent = { hello: 'world' }
|
||
|
sink(name, (args) => {
|
||
|
is(args[0], parent)
|
||
|
is(args[1], msg)
|
||
|
end()
|
||
|
})
|
||
|
const instance = pino({ level: name })
|
||
|
const child = instance.child(parent)
|
||
|
child[name](msg)
|
||
|
})
|
||
|
|
||
|
test('child-child logger for level ' + name, ({ end, is }) => {
|
||
|
var msg = 'hello world'
|
||
|
var grandParent = { hello: 'world' }
|
||
|
var parent = { hello: 'you' }
|
||
|
sink(name, (args) => {
|
||
|
is(args[0], grandParent)
|
||
|
is(args[1], parent)
|
||
|
is(args[2], msg)
|
||
|
end()
|
||
|
})
|
||
|
const instance = pino({ level: name })
|
||
|
const child = instance.child(grandParent).child(parent)
|
||
|
child[name](msg)
|
||
|
})
|
||
|
}
|
||
|
|
||
|
function consoleMethodTest (level, method) {
|
||
|
if (!method) method = level
|
||
|
test('pino().' + level + ' uses console.' + method, ({ end, is }) => {
|
||
|
sink(method, (args) => {
|
||
|
is(args[0], 'test')
|
||
|
end()
|
||
|
})
|
||
|
const instance = require('../browser')({ level: level })
|
||
|
instance[level]('test')
|
||
|
})
|
||
|
}
|
||
|
|
||
|
function absentConsoleMethodTest (method, fallback) {
|
||
|
test('in absence of console.' + method + ', console.' + fallback + ' is used', ({ end, is }) => {
|
||
|
var fn = console[method]
|
||
|
console[method] = undefined
|
||
|
sink(fallback, function (args) {
|
||
|
is(args[0], 'test')
|
||
|
end()
|
||
|
console[method] = fn
|
||
|
})
|
||
|
const instance = require('../browser')({ level: method })
|
||
|
instance[method]('test')
|
||
|
})
|
||
|
}
|
||
|
|
||
|
function isFunc (fn) { return typeof fn === 'function' }
|
||
|
function fnName (fn) {
|
||
|
var rx = /^\s*function\s*([^(]*)/i
|
||
|
var match = rx.exec(fn)
|
||
|
return match && match[1]
|
||
|
}
|
||
|
function sink (method, fn) {
|
||
|
if (method === 'fatal') method = 'error'
|
||
|
var orig = console[method]
|
||
|
console[method] = function () {
|
||
|
console[method] = orig
|
||
|
fn(Array.prototype.slice.call(arguments))
|
||
|
}
|
||
|
}
|