node-logfmt
"logfmt" is the name for a key value logging convention we've adopted at Heroku.
This library is for both converting lines in logfmt format to objects and
for logging objects to a stream in logfmt format.
It provides a logfmt parser, a logging facility,
and both streaming and non-streaming body parsers for express.
You should use this library if you're trying to write structured logs or
if you're consuming them (especially if you're writing a logplex drain).
install
npm install logfmt
use
var logfmt = require('logfmt');
parser
logfmt.parse(string)
var logfmt = require('logfmt');
logfmt.parse("foo=bar a=14 baz=\"hello kitty\" cool%story=bro f %^asdf code=H12")
The only conversions are from the strings true
and false
to their proper boolean counterparts.
We cannot arbitrarily convert numbers because we will drop precision for numbers that require more than 32 bits to represent them.
logging
logfmt.log(object, [stream])
Defaults to logging to process.stdout
var logfmt = require('logfmt');
logfmt.log({ "foo": "bar", "a": 14, baz: 'hello kitty'})
You can have multiple, isolated logfmts by using new
.
var logfmt = require('logfmt');
var errorLogger = new logfmt;
errorLogger.stream = process.stderr
logfmt.log({hello: 'stdout'});
errorLogger.log({hello: 'stderr'});
logfmt.namespace(object)
Returns a new logfmt
with object's data included in every log
call.
var logfmt = require('logfmt').namespace({app: 'logfmt'});
logfmt.log({ "foo": "bar", "a": 14, baz: 'hello kitty'})
logfmt.log({})
logfmt.log({hello: 'world'})
logfmt.time([label], [data], [callback(logger)])
logger.log([data], [stream])
Log how long something takes.
label
: optional name for the milliseconds key (defaults to elapsed
)data
: optional extra data to include with every call to logger.log
logger
: object passed to callback with a log
function
No args defaults to elapsed=<milliseconds>ms
logfmt.time(function(logger){
logger.log();
})
If you don't pass in callback you get the logger returned.
var logger = logfmt.time();
logger.log();
String label
changes the key to <string>=<milliseconds>ms
logfmt.time('time', function(logger){
logger.log();
logger.log();
})
Data can be passed to logger.log
logfmt.time(function(logger){
logger.log({foo: 'bar'});
})
Data can also be passed to logfmt.time
and will persist
across calls to logger.log
logfmt.time('thing', {foo: 'bar'}, function(logger){
logger.log({at: 'function.start'});
logger.log({at: 'function.end'});
})
You do not need a label
to pass data to logfmt.time
logfmt.time({foo: 'bar'}, function(logger){
logger.log({at: 'function'});
})
time
works with namespace
as expected
var logfmt = require('logfmt').namespace({app: 'logfmt'})
logfmt.time({foo: 'bar'}, function(logger){
logger.log({at: 'function'});
})
customizing logging location
logfmt.log()
and logger.log()
Accepts as 2nd argument anything that responds to write(string)
var logfmt = require('logfmt');
logfmt.log({ "foo": "bar", "a": 14, baz: 'hello kitty'}, process.stderr)
Overwrite the default global location by setting logfmt.stream
var logfmt = require('logfmt');
logfmt.stream = process.stderr
logfmt.log({ "foo": "bar", "a": 14, baz: 'hello kitty'})
logging errors
logfmt.error()
Accepts a Javascript Error
object and converts it to logfmt format
var logfmt = require('logfmmt');
logfmt.error(new Error('test error'));
express/restify logging middleware
app.use(logfmt.requestLogger());
logfmt.requestLogger([options], [formatter(req, res)])
If no formatter is supplied it will default to logfmt.requestLogger.commonFormatter
which is based
on having similiar fields to the Apache Common Log format.
Valid Options:
immediate
: log before call to next()
(ie: before the request finishes)elapsed
: renames the elapsed
key to a key of your choice when in
non-immediate mode
Defaults to immediate: true
and elapsed: 'elapsed'
app.use(logfmt.requestLogger({immediate: true}, function(req, res){
return {
method: req.method
}
}));
app.use(logfmt.requestLogger({elapsed: 'request.time'}, function(req, res){
return {
"request.method": req.method
}
}));
formatter(req, res)
A formatter takes the request and response and returns a JSON object for logfmt.log
app.use(logfmt.requestLogger(function(req, res){
return {
method: req.method
}
}));
It's always possible to piggyback on top of the commonFormatter
app.use(logfmt.requestLogger(function(req, res){
var data = logfmt.requestLogger.commonFormatter(req, res)
return {
ip: data.ip,
time: data.time,
foo: 'bar'
};
}));
express/restify parsing middleware
app.use(logfmt.bodyParserStream());
app.use(logfmt.bodyParser());
Streaming
logfmt.bodyParserStream([opts])
Valid Options:
contentType
: defaults to application/logplex-1
If you use the logfmt.bodyParserStream()
for a body parser,
you will have a req.body
that is a readable stream.
Pipes FTW:
var app = require('express')();
var http = require('http');
var through = require('through');
var logfmt = require('logfmt');
app.use(logfmt.bodyParserStream());
app.post('/logs', function(req, res){
if(!req.body) return res.send('OK');
req.body.pipe(through(function(line){
console.dir(line);
}))
res.send('OK');
})
http.createServer(app).listen(3000);
Or you can just use the readable
event:
var app = require('express')();
var http = require('http');
var logfmt = require('logfmt');
app.use(logfmt.bodyParserStream());
app.post('/logs', function(req, res){
req.body.on('readable', function(){
var parsedLine = req.body.read();
if(parsedLine) console.log(parsedLine);
else res.send('OK');
})
})
http.createServer(app).listen(3000);
Non-Streaming
logfmt.bodyParser([opts])
Valid Options:
contentType
: defaults to application/logplex-1
If you use the logfmt.bodyParser()
for a body parser,
you will have a req.body
that is an array of objects.
var logfmt = require('logfmt');
app.use(logfmt.bodyParser());
app.post('/logs', function(req, res){
console.log('BODY: ' + JSON.stringify(req.body));
req.body.forEach(function(data){
console.log(data);
});
res.send('OK');
})
http.createServer(app).listen(3000);
test it:
curl -X POST --header 'Content-Type: application/logplex-1' -d "foo=bar a=14 baz=\"hello kitty\" cool%story=bro f %^asdf" http://localhost:3000/logs
command line
logfmt
accepts lines on STDIN and converts them to json
> echo "foo=bar a=14 baz=\"hello kitty\" cool%story=bro f %^asdf" | logfmt
{ "foo": "bar", "a": 14, "baz": "hello kitty", "cool%story": "bro", "f": true, "%^asdf": true }
logfmt -r (reverse)
accepts JSON on STDIN and converts them to logfmt
> echo '{ "foo": "bar", "a": 14, "baz": "hello kitty", "cool%story": "bro", "f": true, "%^asdf": true }' | logfmt -r
foo=bar a=14 baz="hello kitty" cool%story=bro f=true %^asdf=true
> echo "foo=bar a=14 baz=\"hello kitty\" cool%story=bro f %^asdf" | logfmt | logfmt -r | logfmt
{ "foo": "bar", "a": 14, "baz": "hello kitty", "cool%story": "bro", "f": true, "%^asdf": true }