mirror of
https://github.com/grafana/grafana.git
synced 2025-02-20 11:48:34 -06:00
* First pass solution * Refactor solution * Add test coverage, update tests * Fix behaviour for multiple labels, add test for this * Add recordin rules to devenv prometheus * Update devenv/prometheus2 instead of devenv/prometheus * Add newlines * Fix label matching if labels include comma, add test coverage * Refactor * Refactor, simplify
144 lines
5.3 KiB
TypeScript
144 lines
5.3 KiB
TypeScript
import { expandRecordingRules, fixSummariesMetadata, parseSelector } from './language_utils';
|
|
|
|
describe('parseSelector()', () => {
|
|
let parsed;
|
|
|
|
it('returns a clean selector from an empty selector', () => {
|
|
parsed = parseSelector('{}', 1);
|
|
expect(parsed.selector).toBe('{}');
|
|
expect(parsed.labelKeys).toEqual([]);
|
|
});
|
|
|
|
it('throws if selector is broken', () => {
|
|
expect(() => parseSelector('{foo')).toThrow();
|
|
});
|
|
|
|
it('returns the selector sorted by label key', () => {
|
|
parsed = parseSelector('{foo="bar"}');
|
|
expect(parsed.selector).toBe('{foo="bar"}');
|
|
expect(parsed.labelKeys).toEqual(['foo']);
|
|
|
|
parsed = parseSelector('{foo="bar",baz="xx"}');
|
|
expect(parsed.selector).toBe('{baz="xx",foo="bar"}');
|
|
});
|
|
|
|
it('returns a clean selector from an incomplete one', () => {
|
|
parsed = parseSelector('{foo}');
|
|
expect(parsed.selector).toBe('{}');
|
|
|
|
parsed = parseSelector('{foo="bar",baz}');
|
|
expect(parsed.selector).toBe('{foo="bar"}');
|
|
|
|
parsed = parseSelector('{foo="bar",baz="}');
|
|
expect(parsed.selector).toBe('{foo="bar"}');
|
|
|
|
// Cursor in value area counts as incomplete
|
|
parsed = parseSelector('{foo="bar",baz=""}', 16);
|
|
expect(parsed.selector).toBe('{foo="bar"}');
|
|
|
|
parsed = parseSelector('{foo="bar",baz="4"}', 17);
|
|
expect(parsed.selector).toBe('{foo="bar"}');
|
|
});
|
|
|
|
it('throws if not inside a selector', () => {
|
|
expect(() => parseSelector('foo{}', 0)).toThrow();
|
|
expect(() => parseSelector('foo{} + bar{}', 5)).toThrow();
|
|
});
|
|
|
|
it('returns the selector nearest to the cursor offset', () => {
|
|
expect(() => parseSelector('{foo="bar"} + {foo="bar"}', 0)).toThrow();
|
|
|
|
parsed = parseSelector('{foo="bar"} + {foo="bar"}', 1);
|
|
expect(parsed.selector).toBe('{foo="bar"}');
|
|
|
|
parsed = parseSelector('{foo="bar"} + {baz="xx"}', 1);
|
|
expect(parsed.selector).toBe('{foo="bar"}');
|
|
|
|
parsed = parseSelector('{baz="xx"} + {foo="bar"}', 16);
|
|
expect(parsed.selector).toBe('{foo="bar"}');
|
|
});
|
|
|
|
it('returns a selector with metric if metric is given', () => {
|
|
parsed = parseSelector('bar{foo}', 4);
|
|
expect(parsed.selector).toBe('{__name__="bar"}');
|
|
|
|
parsed = parseSelector('baz{foo="bar"}', 13);
|
|
expect(parsed.selector).toBe('{__name__="baz",foo="bar"}');
|
|
|
|
parsed = parseSelector('bar:metric:1m{}', 14);
|
|
expect(parsed.selector).toBe('{__name__="bar:metric:1m"}');
|
|
});
|
|
});
|
|
|
|
describe('fixSummariesMetadata', () => {
|
|
it('returns empty metadata', () => {
|
|
expect(fixSummariesMetadata({})).toEqual({});
|
|
});
|
|
|
|
it('returns unchanged metadata if no summary is present', () => {
|
|
const metadata = {
|
|
foo: [{ type: 'not_a_summary', help: 'foo help' }],
|
|
};
|
|
expect(fixSummariesMetadata(metadata)).toEqual(metadata);
|
|
});
|
|
|
|
it('returns metadata with added count and sum for a summary', () => {
|
|
const metadata = {
|
|
foo: [{ type: 'not_a_summary', help: 'foo help' }],
|
|
bar: [{ type: 'summary', help: 'bar help' }],
|
|
};
|
|
const expected = {
|
|
foo: [{ type: 'not_a_summary', help: 'foo help' }],
|
|
bar: [{ type: 'summary', help: 'bar help' }],
|
|
bar_count: [{ type: 'counter', help: 'Count of events that have been observed for the base metric (bar help)' }],
|
|
bar_sum: [{ type: 'counter', help: 'Total sum of all observed values for the base metric (bar help)' }],
|
|
};
|
|
expect(fixSummariesMetadata(metadata)).toEqual(expected);
|
|
});
|
|
});
|
|
|
|
describe('expandRecordingRules()', () => {
|
|
it('returns query w/o recording rules as is', () => {
|
|
expect(expandRecordingRules('metric', {})).toBe('metric');
|
|
expect(expandRecordingRules('metric + metric', {})).toBe('metric + metric');
|
|
expect(expandRecordingRules('metric{}', {})).toBe('metric{}');
|
|
});
|
|
|
|
it('does not modify recording rules name in label values', () => {
|
|
expect(expandRecordingRules('{__name__="metric"} + bar', { metric: 'foo', bar: 'super' })).toBe(
|
|
'{__name__="metric"} + super'
|
|
);
|
|
});
|
|
|
|
it('returns query with expanded recording rules', () => {
|
|
expect(expandRecordingRules('metric', { metric: 'foo' })).toBe('foo');
|
|
expect(expandRecordingRules('metric + metric', { metric: 'foo' })).toBe('foo + foo');
|
|
expect(expandRecordingRules('metric{}', { metric: 'foo' })).toBe('foo{}');
|
|
expect(expandRecordingRules('metric[]', { metric: 'foo' })).toBe('foo[]');
|
|
expect(expandRecordingRules('metric + foo', { metric: 'foo', foo: 'bar' })).toBe('foo + bar');
|
|
});
|
|
|
|
it('returns query with labels with expanded recording rules', () => {
|
|
expect(
|
|
expandRecordingRules('metricA{label1="value1"} / metricB{label2="value2"}', { metricA: 'fooA', metricB: 'fooB' })
|
|
).toBe('fooA{label1="value1"} / fooB{label2="value2"}');
|
|
expect(
|
|
expandRecordingRules('metricA{label1="value1",label2="value,2"}', {
|
|
metricA: 'rate(fooA[])',
|
|
})
|
|
).toBe('rate(fooA{label1="value1",label2="value,2"}[])');
|
|
expect(
|
|
expandRecordingRules('metricA{label1="value1"} / metricB{label2="value2"}', {
|
|
metricA: 'rate(fooA[])',
|
|
metricB: 'rate(fooB[])',
|
|
})
|
|
).toBe('rate(fooA{label1="value1"}[])/ rate(fooB{label2="value2"}[])');
|
|
expect(
|
|
expandRecordingRules('metricA{label1="value1",label2="value2"} / metricB{label3="value3"}', {
|
|
metricA: 'rate(fooA[])',
|
|
metricB: 'rate(fooB[])',
|
|
})
|
|
).toBe('rate(fooA{label1="value1",label2="value2"}[])/ rate(fooB{label3="value3"}[])');
|
|
});
|
|
});
|