Node.js Logging Support for Cloud Foundry
Summary
This is a collection of support libraries for node.js applications running on Cloud Foundry that serve two main purposes: It provides (a) means to emit structured application log messages and (b) instrument parts of your application stack to collect request metrics.
For details on the concepts and log formats, please look at the sibling project for java logging support.
Version 2.0 introduced logging without Winston and changed custom fields to be parsed and reported as strings regardless of original type.
Features
- Network logging (http requests) for CloudFoundry
- Custom message logging
- Logging levels
- Can be bound to Winston as transport
Installation
npm install cf-nodejs-logging-support
Usage
Minimal Example
var express = require('express');
var log = require('cf-nodejs-logging-support');
var app = express();
log.setLoggingLevel("info");
app.use(log.logNetwork);
app.get('/', function (req, res) {
req.logMessage("info", "Hello World will be send");
res.send('Hello World');
});
app.listen(3000);
log.logMessage("info", "Server is listening on port %d", 3000);
Other Server Libraries
The logging library defaults to express middleware behaviour, but it can be forced to work with other Server libraries as well:
With restify:
var restify = require('restify');
var log = require('cf-nodejs-logging-support');
var app = restify.createServer();
log.forceLogger("restify");
app.use(log.logNetwork);
With nodejs http:
var log = require("cf-nodejs-logging-support");
const http = require('http');
log.forceLogger("plainhttp");
const server = http.createServer((req, res) => {
log.logNetwork(req, res);
req.logMessage("info", "request bound information:", {
"some": "info"
});
res.end('ok');
});
server.listen(3000);
log.logMessage("info", "Server is listening on port %d", 3000);
Custom Messages
Use the logMessage(...) function to log custom messages with a given logging level. It is also possible to use the standard format placeholders equivalent to the util.format method.
Custom messages may be called on two objects:
- If you want to keep track of a request with custom messages, you want to call req.logMessage(...) so the correct correlation_id and request_id will be added to this log.
- If you want to write request independent custom messages, you want to call log.logMessage(...) so no further context tracking will be added.
Simple message
logMessage("info", "Hello World");
With addtional numeric value
logMessage("info", "Listening on port %d", 5000);
With additional string values
logMessage("info", "This %s a %s", "is", "test");
With custom fields added to custom_fields field. Keep in mind, that the last argument is handled as custom_fields object, if it is an object.
logMessage("info", "Test data %j", {"field" :"value"});
With json object forced to be embedded in to the message (nothing will be added to custom_fields).
logMessage("info", "Test data %j", {"field" :"value"}, {});
Winston Transport
This logging tool can be used in conjunction with Winston. Logging via Winston transport is limited to custom logs. Network activity can not be tracked automatically. Example:
var express = require('express');
var log = require('cf-nodejs-logging-support');
var winston = require('winston');
var app = express();
var logger = new(winston.Logger)({
transports: [log.winstonTransport]
});
app.get('/', function (req, res) {
res.send('Hello World');
});
app.listen(3000);
logger.log("info", "Server is listening on port %d", 3000);
Request correlation_id
In order to get the correlation_id of an request, you can use the following call.
app.get('/', function (req, res) {
var id = req.getCorrelationId();
res.send('Hello World');
});
Human readable output
Setup an output pattern to get a human-readable output instead of json. Use '{{' and '}}' to print log parameters.
log.setLogPattern("{{written_at}} - {{msg}}");
Fixed Values for Network Logging (will impact log parsing if used incorrectly)
Possibility to tailor logs to your needs, you can for example change the msg field for Network-logs to a find them in Human readable format:
log.overrideNetworkField("msg", YOUR_CUSTOM_MSG);
This will replace the value of the previously empty msg field for network logs with YOUR_CUSTOM_MSG.
Sample Apps