2018-12-05 05:18:51 -06:00
|
|
|
import {
|
|
|
|
calculateFieldStats,
|
|
|
|
calculateLogsLabelStats,
|
|
|
|
dedupLogRows,
|
|
|
|
getParser,
|
|
|
|
LogsDedupStrategy,
|
|
|
|
LogsModel,
|
|
|
|
LogsParsers,
|
|
|
|
} from '../logs_model';
|
2018-11-18 03:38:06 -06:00
|
|
|
|
|
|
|
describe('dedupLogRows()', () => {
|
|
|
|
test('should return rows as is when dedup is set to none', () => {
|
|
|
|
const logs = {
|
|
|
|
rows: [
|
|
|
|
{
|
|
|
|
entry: 'WARN test 1.23 on [xxx]',
|
|
|
|
},
|
|
|
|
{
|
|
|
|
entry: 'WARN test 1.23 on [xxx]',
|
|
|
|
},
|
|
|
|
],
|
|
|
|
};
|
|
|
|
expect(dedupLogRows(logs as LogsModel, LogsDedupStrategy.none).rows).toMatchObject(logs.rows);
|
|
|
|
});
|
|
|
|
|
|
|
|
test('should dedup on exact matches', () => {
|
|
|
|
const logs = {
|
|
|
|
rows: [
|
|
|
|
{
|
|
|
|
entry: 'WARN test 1.23 on [xxx]',
|
|
|
|
},
|
|
|
|
{
|
|
|
|
entry: 'WARN test 1.23 on [xxx]',
|
|
|
|
},
|
|
|
|
{
|
|
|
|
entry: 'INFO test 2.44 on [xxx]',
|
|
|
|
},
|
|
|
|
{
|
|
|
|
entry: 'WARN test 1.23 on [xxx]',
|
|
|
|
},
|
|
|
|
],
|
|
|
|
};
|
|
|
|
expect(dedupLogRows(logs as LogsModel, LogsDedupStrategy.exact).rows).toEqual([
|
|
|
|
{
|
|
|
|
duplicates: 1,
|
|
|
|
entry: 'WARN test 1.23 on [xxx]',
|
|
|
|
},
|
|
|
|
{
|
|
|
|
duplicates: 0,
|
|
|
|
entry: 'INFO test 2.44 on [xxx]',
|
|
|
|
},
|
|
|
|
{
|
|
|
|
duplicates: 0,
|
|
|
|
entry: 'WARN test 1.23 on [xxx]',
|
|
|
|
},
|
|
|
|
]);
|
|
|
|
});
|
|
|
|
|
|
|
|
test('should dedup on number matches', () => {
|
|
|
|
const logs = {
|
|
|
|
rows: [
|
|
|
|
{
|
|
|
|
entry: 'WARN test 1.2323423 on [xxx]',
|
|
|
|
},
|
|
|
|
{
|
|
|
|
entry: 'WARN test 1.23 on [xxx]',
|
|
|
|
},
|
|
|
|
{
|
|
|
|
entry: 'INFO test 2.44 on [xxx]',
|
|
|
|
},
|
|
|
|
{
|
|
|
|
entry: 'WARN test 1.23 on [xxx]',
|
|
|
|
},
|
|
|
|
],
|
|
|
|
};
|
|
|
|
expect(dedupLogRows(logs as LogsModel, LogsDedupStrategy.numbers).rows).toEqual([
|
|
|
|
{
|
|
|
|
duplicates: 1,
|
|
|
|
entry: 'WARN test 1.2323423 on [xxx]',
|
|
|
|
},
|
|
|
|
{
|
|
|
|
duplicates: 0,
|
|
|
|
entry: 'INFO test 2.44 on [xxx]',
|
|
|
|
},
|
|
|
|
{
|
|
|
|
duplicates: 0,
|
|
|
|
entry: 'WARN test 1.23 on [xxx]',
|
|
|
|
},
|
|
|
|
]);
|
|
|
|
});
|
|
|
|
|
|
|
|
test('should dedup on signature matches', () => {
|
|
|
|
const logs = {
|
|
|
|
rows: [
|
|
|
|
{
|
|
|
|
entry: 'WARN test 1.2323423 on [xxx]',
|
|
|
|
},
|
|
|
|
{
|
|
|
|
entry: 'WARN test 1.23 on [xxx]',
|
|
|
|
},
|
|
|
|
{
|
|
|
|
entry: 'INFO test 2.44 on [xxx]',
|
|
|
|
},
|
|
|
|
{
|
|
|
|
entry: 'WARN test 1.23 on [xxx]',
|
|
|
|
},
|
|
|
|
],
|
|
|
|
};
|
|
|
|
expect(dedupLogRows(logs as LogsModel, LogsDedupStrategy.signature).rows).toEqual([
|
|
|
|
{
|
|
|
|
duplicates: 3,
|
|
|
|
entry: 'WARN test 1.2323423 on [xxx]',
|
|
|
|
},
|
|
|
|
]);
|
|
|
|
});
|
|
|
|
});
|
2018-12-03 10:00:50 -06:00
|
|
|
|
2018-12-05 05:18:51 -06:00
|
|
|
describe('calculateFieldStats()', () => {
|
|
|
|
test('should return no stats for empty rows', () => {
|
|
|
|
expect(calculateFieldStats([], /foo=(.*)/)).toEqual([]);
|
|
|
|
});
|
|
|
|
|
|
|
|
test('should return no stats if extractor does not match', () => {
|
|
|
|
const rows = [
|
|
|
|
{
|
|
|
|
entry: 'foo=bar',
|
|
|
|
},
|
|
|
|
];
|
|
|
|
|
|
|
|
expect(calculateFieldStats(rows as any, /baz=(.*)/)).toEqual([]);
|
|
|
|
});
|
|
|
|
|
|
|
|
test('should return stats for found field', () => {
|
|
|
|
const rows = [
|
|
|
|
{
|
|
|
|
entry: 'foo="42 + 1"',
|
|
|
|
},
|
|
|
|
{
|
|
|
|
entry: 'foo=503 baz=foo',
|
|
|
|
},
|
|
|
|
{
|
|
|
|
entry: 'foo="42 + 1"',
|
|
|
|
},
|
|
|
|
{
|
|
|
|
entry: 't=2018-12-05T07:44:59+0000 foo=503',
|
|
|
|
},
|
|
|
|
];
|
|
|
|
|
|
|
|
expect(calculateFieldStats(rows as any, /foo=("[^"]*"|\S+)/)).toMatchObject([
|
|
|
|
{
|
|
|
|
value: '"42 + 1"',
|
|
|
|
count: 2,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
value: '503',
|
|
|
|
count: 2,
|
|
|
|
},
|
|
|
|
]);
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
2018-12-03 10:00:50 -06:00
|
|
|
describe('calculateLogsLabelStats()', () => {
|
|
|
|
test('should return no stats for empty rows', () => {
|
|
|
|
expect(calculateLogsLabelStats([], '')).toEqual([]);
|
|
|
|
});
|
|
|
|
|
|
|
|
test('should return no stats of label is not found', () => {
|
|
|
|
const rows = [
|
|
|
|
{
|
|
|
|
entry: 'foo 1',
|
|
|
|
labels: {
|
|
|
|
foo: 'bar',
|
|
|
|
},
|
|
|
|
},
|
|
|
|
];
|
|
|
|
|
|
|
|
expect(calculateLogsLabelStats(rows as any, 'baz')).toEqual([]);
|
|
|
|
});
|
|
|
|
|
|
|
|
test('should return stats for found labels', () => {
|
|
|
|
const rows = [
|
|
|
|
{
|
|
|
|
entry: 'foo 1',
|
|
|
|
labels: {
|
|
|
|
foo: 'bar',
|
|
|
|
},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
entry: 'foo 0',
|
|
|
|
labels: {
|
|
|
|
foo: 'xxx',
|
|
|
|
},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
entry: 'foo 2',
|
|
|
|
labels: {
|
|
|
|
foo: 'bar',
|
|
|
|
},
|
|
|
|
},
|
|
|
|
];
|
|
|
|
|
|
|
|
expect(calculateLogsLabelStats(rows as any, 'foo')).toMatchObject([
|
|
|
|
{
|
|
|
|
value: 'bar',
|
|
|
|
count: 2,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
value: 'xxx',
|
|
|
|
count: 1,
|
|
|
|
},
|
|
|
|
]);
|
|
|
|
});
|
|
|
|
});
|
2018-12-05 05:18:51 -06:00
|
|
|
|
|
|
|
describe('getParser()', () => {
|
|
|
|
test('should return no parser on empty line', () => {
|
|
|
|
expect(getParser('')).toBeUndefined();
|
|
|
|
});
|
|
|
|
|
|
|
|
test('should return no parser on unknown line pattern', () => {
|
|
|
|
expect(getParser('To Be or not to be')).toBeUndefined();
|
|
|
|
});
|
|
|
|
|
|
|
|
test('should return logfmt parser on key value patterns', () => {
|
|
|
|
expect(getParser('foo=bar baz="41 + 1')).toEqual(LogsParsers.logfmt);
|
|
|
|
});
|
|
|
|
|
|
|
|
test('should return JSON parser on JSON log lines', () => {
|
|
|
|
// TODO implement other JSON value types than string
|
|
|
|
expect(getParser('{"foo": "bar", "baz": "41 + 1"}')).toEqual(LogsParsers.JSON);
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
describe('LogsParsers', () => {
|
|
|
|
describe('logfmt', () => {
|
|
|
|
const parser = LogsParsers.logfmt;
|
|
|
|
|
|
|
|
test('should detect format', () => {
|
|
|
|
expect(parser.test('foo')).toBeFalsy();
|
|
|
|
expect(parser.test('foo=bar')).toBeTruthy();
|
|
|
|
});
|
|
|
|
|
2018-12-10 12:05:57 -06:00
|
|
|
test('should return parsed fields', () => {
|
|
|
|
expect(parser.getFields('foo=bar baz="42 + 1"')).toEqual(['foo=bar', 'baz="42 + 1"']);
|
|
|
|
});
|
|
|
|
|
|
|
|
test('should return label for field', () => {
|
|
|
|
expect(parser.getLabelFromField('foo=bar')).toBe('foo');
|
|
|
|
});
|
|
|
|
|
|
|
|
test('should return value for field', () => {
|
|
|
|
expect(parser.getValueFromField('foo=bar')).toBe('bar');
|
2018-12-05 05:18:51 -06:00
|
|
|
});
|
|
|
|
|
|
|
|
test('should build a valid value matcher', () => {
|
|
|
|
const matcher = parser.buildMatcher('foo');
|
|
|
|
const match = 'foo=bar'.match(matcher);
|
|
|
|
expect(match).toBeDefined();
|
|
|
|
expect(match[1]).toBe('bar');
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
describe('JSON', () => {
|
|
|
|
const parser = LogsParsers.JSON;
|
|
|
|
|
|
|
|
test('should detect format', () => {
|
|
|
|
expect(parser.test('foo')).toBeFalsy();
|
|
|
|
expect(parser.test('{"foo":"bar"}')).toBeTruthy();
|
|
|
|
});
|
|
|
|
|
2018-12-10 12:05:57 -06:00
|
|
|
test('should return parsed fields', () => {
|
|
|
|
expect(parser.getFields('{ "foo" : "bar", "baz" : 42 }')).toEqual(['"foo" : "bar"', '"baz" : 42']);
|
2018-12-05 05:18:51 -06:00
|
|
|
});
|
|
|
|
|
2018-12-10 12:05:57 -06:00
|
|
|
test('should return parsed fields for nested quotes', () => {
|
|
|
|
expect(parser.getFields(`{"foo":"bar: '[value=\\"42\\"]'"}`)).toEqual([`"foo":"bar: '[value=\\"42\\"]'"`]);
|
|
|
|
});
|
|
|
|
|
|
|
|
test('should return label for field', () => {
|
|
|
|
expect(parser.getLabelFromField('"foo" : "bar"')).toBe('foo');
|
|
|
|
});
|
|
|
|
|
|
|
|
test('should return value for field', () => {
|
|
|
|
expect(parser.getValueFromField('"foo" : "bar"')).toBe('"bar"');
|
|
|
|
expect(parser.getValueFromField('"foo" : 42')).toBe('42');
|
|
|
|
expect(parser.getValueFromField('"foo" : 42.1')).toBe('42.1');
|
|
|
|
});
|
|
|
|
|
|
|
|
test('should build a valid value matcher for strings', () => {
|
2018-12-05 05:18:51 -06:00
|
|
|
const matcher = parser.buildMatcher('foo');
|
|
|
|
const match = '{"foo":"bar"}'.match(matcher);
|
|
|
|
expect(match).toBeDefined();
|
|
|
|
expect(match[1]).toBe('bar');
|
|
|
|
});
|
2018-12-10 12:05:57 -06:00
|
|
|
|
|
|
|
test('should build a valid value matcher for integers', () => {
|
|
|
|
const matcher = parser.buildMatcher('foo');
|
|
|
|
const match = '{"foo":42.1}'.match(matcher);
|
|
|
|
expect(match).toBeDefined();
|
|
|
|
expect(match[1]).toBe('42.1');
|
|
|
|
});
|
2018-12-05 05:18:51 -06:00
|
|
|
});
|
|
|
|
});
|