log4js-node/test/vows/layouts-test.js
e-cloud 3e4f3a66f3 style: format test code
a error occur in `loggin-test.js` - 'invalid configuration'.

It may reveal some internal defects of this library.

Another problem occurs in `configureNoLevels-test.js`. I mark a
**todo** there.
2016-12-12 22:23:39 +08:00

372 lines
12 KiB
JavaScript

'use strict';
const vows = require('vows');
const assert = require('assert');
const os = require('os');
const semver = require('semver');
const EOL = os.EOL || '\n';
// used for patternLayout tests.
function test(args, pattern, value) {
const layout = args[0];
const event = args[1];
const tokens = args[2];
assert.equal(layout(pattern, tokens)(event), value);
}
vows.describe('log4js layouts').addBatch({
colouredLayout: {
topic: function () {
return require('../../lib/layouts').colouredLayout;
},
'should apply level colour codes to output': function (layout) {
const output = layout({
data: ['nonsense'],
startTime: new Date(2010, 11, 5, 14, 18, 30, 45),
categoryName: 'cheese',
level: {
toString: function () {
return 'ERROR';
}
}
});
assert.equal(output, '\x1B[31m[2010-12-05 14:18:30.045] [ERROR] cheese - \x1B[39mnonsense');
},
'should support the console.log format for the message': function (layout) {
const output = layout({
data: ['thing %d', 2],
startTime: new Date(2010, 11, 5, 14, 18, 30, 45),
categoryName: 'cheese',
level: {
toString: function () {
return 'ERROR';
}
}
});
assert.equal(output, '\x1B[31m[2010-12-05 14:18:30.045] [ERROR] cheese - \x1B[39mthing 2');
}
},
messagePassThroughLayout: {
topic: function () {
return require('../../lib/layouts').messagePassThroughLayout;
},
'should take a logevent and output only the message': function (layout) {
assert.equal(layout({
data: ['nonsense'],
startTime: new Date(2010, 11, 5, 14, 18, 30, 45),
categoryName: 'cheese',
level: {
colour: 'green',
toString: function () {
return 'ERROR';
}
}
}), 'nonsense');
},
'should support the console.log format for the message': function (layout) {
assert.equal(layout({
data: ['thing %d', 1, 'cheese'],
startTime: new Date(2010, 11, 5, 14, 18, 30, 45),
categoryName: 'cheese',
level: {
colour: 'green',
toString: function () {
return 'ERROR';
}
}
}), 'thing 1 cheese');
},
'should output the first item even if it is not a string': function (layout) {
assert.equal(layout({
data: [{ thing: 1 }],
startTime: new Date(2010, 11, 5, 14, 18, 30, 45),
categoryName: 'cheese',
level: {
colour: 'green',
toString: function () {
return 'ERROR';
}
}
}), '{ thing: 1 }');
},
'should print the stacks of a passed error objects': function (layout) {
assert.isArray(
layout({
data: [new Error()],
startTime: new Date(2010, 11, 5, 14, 18, 30, 45),
categoryName: 'cheese',
level: {
colour: 'green',
toString: function () {
return 'ERROR';
}
}
}).match(
new RegExp(`${/Error\s+at Object\..*\s+/.source}${/\((.*)test[\\/]vows[\\/]layouts-test\.js/.source}${/:\d+:\d+\)\s+at runTest/.source}`
)
),
'regexp did not return a match'
);
},
'with passed augmented errors': {
topic: function (layout) {
const e = new Error('My Unique Error Message');
e.augmented = 'My Unique attribute value';
e.augObj = { at1: 'at2' };
return layout({
data: [e],
startTime: new Date(2010, 11, 5, 14, 18, 30, 45),
categoryName: 'cheese',
level: {
colour: 'green',
toString: function () {
return 'ERROR';
}
}
});
},
'should print error the contained error message': function (layoutOutput) {
const m = layoutOutput.match(/Error: My Unique Error Message/);
assert.isArray(m);
},
'should print error augmented string attributes': function (layoutOutput) {
const m = layoutOutput.match(/augmented:\s'My Unique attribute value'/);
assert.isArray(m);
},
'should print error augmented object attributes': function (layoutOutput) {
const m = layoutOutput.match(/augObj:\s\{ at1: 'at2' \}/);
assert.isArray(m);
}
}
},
basicLayout: {
topic: function () {
const layout = require('../../lib/layouts').basicLayout;
const event = {
data: ['this is a test'],
startTime: new Date(2010, 11, 5, 14, 18, 30, 45),
categoryName: 'tests',
level: {
toString: function () {
return 'DEBUG';
}
}
};
return [layout, event];
},
'should take a logevent and output a formatted string': function (args) {
const layout = args[0];
const event = args[1];
assert.equal(layout(event), '[2010-12-05 14:18:30.045] [DEBUG] tests - this is a test');
},
'should output a stacktrace, message if the event has an error attached': function (args) {
let i;
const layout = args[0];
const event = args[1];
const error = new Error('Some made-up error');
const stack = error.stack.split(/\n/);
event.data = ['this is a test', error];
const output = layout(event);
const lines = output.split(/\n/);
if (semver.satisfies(process.version, '>=6')) {
assert.equal(lines.length, stack.length);
assert.equal(
lines[0],
'[2010-12-05 14:18:30.045] [DEBUG] tests - this is a test Error: Some made-up error'
);
for (i = 1; i < stack.length; i++) {
assert.equal(lines[i], stack[i]);
}
} else {
assert.equal(lines.length - 1, stack.length);
assert.equal(
lines[0],
'[2010-12-05 14:18:30.045] [DEBUG] tests - this is a test [Error: Some made-up error]'
);
for (i = 1; i < stack.length; i++) {
assert.equal(lines[i + 2], stack[i + 1]);
}
}
},
'should output any extra data in the log event as util.inspect strings': function (args) {
const layout = args[0];
const event = args[1];
event.data = [
'this is a test', {
name: 'Cheese',
message: 'Gorgonzola smells.'
}
];
const output = layout(event);
assert.equal(
output,
'[2010-12-05 14:18:30.045] [DEBUG] tests - this is a test ' +
"{ name: 'Cheese', message: 'Gorgonzola smells.' }"
);
}
},
patternLayout: {
topic: function () {
const event = {
data: ['this is a test'],
startTime: new Date('2010-12-05T14:18:30.045Z'), // new Date(2010, 11, 5, 14, 18, 30, 45),
categoryName: 'multiple.levels.of.tests',
level: {
toString: function () {
return 'DEBUG';
}
}
};
const layout = require('../../lib/layouts').patternLayout;
const tokens = {
testString: 'testStringToken',
testFunction: function () {
return 'testFunctionToken';
},
fnThatUsesLogEvent: function (logEvent) {
return logEvent.level.toString();
}
};
// override getTimezoneOffset
event.startTime.getTimezoneOffset = function () {
return 0;
};
return [layout, event, tokens];
},
'should default to "time logLevel loggerName - message"': function (args) {
test(args, null, `14:18:30 DEBUG multiple.levels.of.tests - this is a test${EOL}`);
},
'%r should output time only': function (args) {
test(args, '%r', '14:18:30');
},
'%p should output the log level': function (args) {
test(args, '%p', 'DEBUG');
},
'%c should output the log category': function (args) {
test(args, '%c', 'multiple.levels.of.tests');
},
'%m should output the log data': function (args) {
test(args, '%m', 'this is a test');
},
'%n should output a new line': function (args) {
test(args, '%n', EOL);
},
'%h should output hostname': function (args) {
test(args, '%h', os.hostname().toString());
},
'%z should output pid': function (args) {
test(args, '%z', process.pid);
},
'%c should handle category names like java-style package names': function (args) {
test(args, '%c{1}', 'tests');
test(args, '%c{2}', 'of.tests');
test(args, '%c{3}', 'levels.of.tests');
test(args, '%c{4}', 'multiple.levels.of.tests');
test(args, '%c{5}', 'multiple.levels.of.tests');
test(args, '%c{99}', 'multiple.levels.of.tests');
},
'%d should output the date in ISO8601 format': function (args) {
test(args, '%d', '2010-12-05 14:18:30.045');
},
'%d should allow for format specification': function (args) {
test(args, '%d{ISO8601_WITH_TZ_OFFSET}', '2010-12-05T14:18:30.045-0000');
test(args, '%d{ISO8601}', '2010-12-05 14:18:30.045');
test(args, '%d{ABSOLUTE}', '14:18:30.045');
test(args, '%d{DATE}', '05 12 2010 14:18:30.045');
test(args, '%d{yy MM dd hh mm ss}', '10 12 05 14 18 30');
test(args, '%d{yyyy MM dd}', '2010 12 05');
test(args, '%d{yyyy MM dd hh mm ss SSS}', '2010 12 05 14 18 30 045');
},
'%% should output %': function (args) {
test(args, '%%', '%');
},
'should output anything not preceded by % as literal': function (args) {
test(args, 'blah blah blah', 'blah blah blah');
},
'should output the original string if no replacer matches the token': function (args) {
test(args, '%a{3}', 'a{3}');
},
'should handle complicated patterns': function (args) {
test(args,
'%m%n %c{2} at %d{ABSOLUTE} cheese %p%n',
`this is a test${EOL} of.tests at 14:18:30.045 cheese DEBUG${EOL}`
);
},
'should truncate fields if specified': function (args) {
test(args, '%.4m', 'this');
test(args, '%.7m', 'this is');
test(args, '%.9m', 'this is a');
test(args, '%.14m', 'this is a test');
test(args, '%.2919102m', 'this is a test');
},
'should pad fields if specified': function (args) {
test(args, '%10p', ' DEBUG');
test(args, '%8p', ' DEBUG');
test(args, '%6p', ' DEBUG');
test(args, '%4p', 'DEBUG');
test(args, '%-4p', 'DEBUG');
test(args, '%-6p', 'DEBUG ');
test(args, '%-8p', 'DEBUG ');
test(args, '%-10p', 'DEBUG ');
},
'%[%r%] should output colored time': function (args) {
test(args, '%[%r%]', '\x1B[36m14:18:30\x1B[39m');
},
'%x{testString} should output the string stored in tokens': function (args) {
test(args, '%x{testString}', 'testStringToken');
},
'%x{testFunction} should output the result of the function stored in tokens': function (args) {
test(args, '%x{testFunction}', 'testFunctionToken');
},
'%x{doesNotExist} should output the string stored in tokens': function (args) {
test(args, '%x{doesNotExist}', 'null');
},
'%x{fnThatUsesLogEvent} should be able to use the logEvent': function (args) {
test(args, '%x{fnThatUsesLogEvent}', 'DEBUG');
},
'%x should output the string stored in tokens': function (args) {
test(args, '%x', 'null');
}
},
'layout makers': {
topic: require('../../lib/layouts'),
'should have a maker for each layout': function (layouts) {
assert.ok(layouts.layout('messagePassThrough'));
assert.ok(layouts.layout('basic'));
assert.ok(layouts.layout('colored'));
assert.ok(layouts.layout('coloured'));
assert.ok(layouts.layout('pattern'));
}
},
'add layout': {
topic: require('../../lib/layouts'),
'should be able to add a layout': function (layouts) {
layouts.addLayout('test_layout', (config) => {
assert.equal(config, 'test_config');
return function (logEvent) {
return `TEST LAYOUT >${logEvent.data}`;
};
});
const serializer = layouts.layout('test_layout', 'test_config');
assert.ok(serializer);
assert.equal(serializer({ data: 'INPUT' }), 'TEST LAYOUT >INPUT');
}
}
}).export(module);