| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238 |
- // @ts-ignore
- import Plain from 'slate-plain-serializer';
- import LanguageProvider, { LABEL_REFRESH_INTERVAL, LokiHistoryItem, rangeToParams } from './language_provider';
- import { AbsoluteTimeRange } from '@grafana/data';
- import { advanceTo, clear, advanceBy } from 'jest-date-mock';
- import { beforeEach } from 'test/lib/common';
- import { DataSourceApi } from '@grafana/ui';
- import { TypeaheadInput } from '../../../types';
- import { makeMockLokiDatasource } from './mocks';
- describe('Language completion provider', () => {
- const datasource = makeMockLokiDatasource({});
- const rangeMock: AbsoluteTimeRange = {
- from: 1560153109000,
- to: 1560163909000,
- };
- describe('empty query suggestions', () => {
- it('returns no suggestions on empty context', () => {
- const instance = new LanguageProvider(datasource);
- const value = Plain.deserialize('');
- const result = instance.provideCompletionItems({ text: '', prefix: '', value, wrapperClasses: [] });
- expect(result.context).toBeUndefined();
- expect(result.refresher).toBeUndefined();
- expect(result.suggestions.length).toEqual(0);
- });
- it('returns default suggestions with history on empty context when history was provided', () => {
- const instance = new LanguageProvider(datasource);
- const value = Plain.deserialize('');
- const history: LokiHistoryItem[] = [
- {
- query: { refId: '1', expr: '{app="foo"}' },
- ts: 1,
- },
- ];
- const result = instance.provideCompletionItems(
- { text: '', prefix: '', value, wrapperClasses: [] },
- { history, absoluteRange: rangeMock }
- );
- expect(result.context).toBeUndefined();
- expect(result.refresher).toBeUndefined();
- expect(result.suggestions).toMatchObject([
- {
- label: 'History',
- items: [
- {
- label: '{app="foo"}',
- },
- ],
- },
- ]);
- });
- it('returns no suggestions within regexp', () => {
- const instance = new LanguageProvider(datasource);
- const input = createTypeaheadInput('{} ()', '', undefined, 4, []);
- const history: LokiHistoryItem[] = [
- {
- query: { refId: '1', expr: '{app="foo"}' },
- ts: 1,
- },
- ];
- const result = instance.provideCompletionItems(input, { history });
- expect(result.context).toBeUndefined();
- expect(result.refresher).toBeUndefined();
- expect(result.suggestions.length).toEqual(0);
- });
- });
- describe('label suggestions', () => {
- it('returns default label suggestions on label context', () => {
- const instance = new LanguageProvider(datasource);
- const input = createTypeaheadInput('{}', '');
- const result = instance.provideCompletionItems(input, { absoluteRange: rangeMock });
- expect(result.context).toBe('context-labels');
- expect(result.suggestions).toEqual([{ items: [{ label: 'job' }, { label: 'namespace' }], label: 'Labels' }]);
- });
- it('returns label suggestions from Loki', async () => {
- const datasource = makeMockLokiDatasource({ label1: [], label2: [] });
- const provider = await getLanguageProvider(datasource);
- const input = createTypeaheadInput('{}', '');
- const result = provider.provideCompletionItems(input, { absoluteRange: rangeMock });
- expect(result.context).toBe('context-labels');
- expect(result.suggestions).toEqual([{ items: [{ label: 'label1' }, { label: 'label2' }], label: 'Labels' }]);
- });
- it('returns label values suggestions from Loki', async () => {
- const datasource = makeMockLokiDatasource({ label1: ['label1_val1', 'label1_val2'], label2: [] });
- const provider = await getLanguageProvider(datasource);
- const input = createTypeaheadInput('{label1=}', '=', 'label1');
- let result = provider.provideCompletionItems(input, { absoluteRange: rangeMock });
- // The values for label are loaded adhoc and there is a promise returned that we have to wait for
- expect(result.refresher).toBeDefined();
- await result.refresher;
- result = provider.provideCompletionItems(input, { absoluteRange: rangeMock });
- expect(result.context).toBe('context-label-values');
- expect(result.suggestions).toEqual([
- { items: [{ label: 'label1_val1' }, { label: 'label1_val2' }], label: 'Label values for "label1"' },
- ]);
- });
- });
- });
- describe('Request URL', () => {
- it('should contain range params', async () => {
- const rangeMock: AbsoluteTimeRange = {
- from: 1560153109000,
- to: 1560163909000,
- };
- const datasourceWithLabels = makeMockLokiDatasource({ other: [] });
- const datasourceSpy = jest.spyOn(datasourceWithLabels as any, 'metadataRequest');
- const instance = new LanguageProvider(datasourceWithLabels, { initialRange: rangeMock });
- await instance.refreshLogLabels(rangeMock, true);
- const expectedUrl = '/api/prom/label';
- expect(datasourceSpy).toHaveBeenCalledWith(expectedUrl, rangeToParams(rangeMock));
- });
- });
- describe('Query imports', () => {
- const datasource = makeMockLokiDatasource({});
- const rangeMock: AbsoluteTimeRange = {
- from: 1560153109000,
- to: 1560163909000,
- };
- it('returns empty queries for unknown origin datasource', async () => {
- const instance = new LanguageProvider(datasource, { initialRange: rangeMock });
- const result = await instance.importQueries([{ refId: 'bar', expr: 'foo' }], 'unknown');
- expect(result).toEqual([{ refId: 'bar', expr: '' }]);
- });
- describe('prometheus query imports', () => {
- it('returns empty query from metric-only query', async () => {
- const instance = new LanguageProvider(datasource, { initialRange: rangeMock });
- const result = await instance.importPrometheusQuery('foo');
- expect(result).toEqual('');
- });
- it('returns empty query from selector query if label is not available', async () => {
- const datasourceWithLabels = makeMockLokiDatasource({ other: [] });
- const instance = new LanguageProvider(datasourceWithLabels, { initialRange: rangeMock });
- const result = await instance.importPrometheusQuery('{foo="bar"}');
- expect(result).toEqual('{}');
- });
- it('returns selector query from selector query with common labels', async () => {
- const datasourceWithLabels = makeMockLokiDatasource({ foo: [] });
- const instance = new LanguageProvider(datasourceWithLabels, { initialRange: rangeMock });
- const result = await instance.importPrometheusQuery('metric{foo="bar",baz="42"}');
- expect(result).toEqual('{foo="bar"}');
- });
- it('returns selector query from selector query with all labels if logging label list is empty', async () => {
- const datasourceWithLabels = makeMockLokiDatasource({});
- const instance = new LanguageProvider(datasourceWithLabels, { initialRange: rangeMock });
- const result = await instance.importPrometheusQuery('metric{foo="bar",baz="42"}');
- expect(result).toEqual('{baz="42",foo="bar"}');
- });
- });
- });
- describe('Labels refresh', () => {
- const datasource = makeMockLokiDatasource({});
- const instance = new LanguageProvider(datasource);
- const rangeMock: AbsoluteTimeRange = {
- from: 1560153109000,
- to: 1560163909000,
- };
- beforeEach(() => {
- instance.fetchLogLabels = jest.fn();
- });
- afterEach(() => {
- jest.clearAllMocks();
- clear();
- });
- it("should not refresh labels if refresh interval hasn't passed", () => {
- advanceTo(new Date(2019, 1, 1, 0, 0, 0));
- instance.logLabelFetchTs = Date.now();
- advanceBy(LABEL_REFRESH_INTERVAL / 2);
- instance.refreshLogLabels(rangeMock);
- expect(instance.fetchLogLabels).not.toBeCalled();
- });
- it('should refresh labels if refresh interval passed', () => {
- advanceTo(new Date(2019, 1, 1, 0, 0, 0));
- instance.logLabelFetchTs = Date.now();
- advanceBy(LABEL_REFRESH_INTERVAL + 1);
- instance.refreshLogLabels(rangeMock);
- expect(instance.fetchLogLabels).toBeCalled();
- });
- });
- async function getLanguageProvider(datasource: DataSourceApi) {
- const instance = new LanguageProvider(datasource);
- instance.initialRange = {
- from: Date.now() - 10000,
- to: Date.now(),
- };
- await instance.start();
- return instance;
- }
- /**
- * @param value Value of the full input
- * @param text Last piece of text (not sure but in case of {label=} this would be just '=')
- * @param labelKey Label by which to search for values. Cutting corners a bit here as this should be inferred from value
- */
- function createTypeaheadInput(
- value: string,
- text: string,
- labelKey?: string,
- anchorOffset?: number,
- wrapperClasses?: string[]
- ): TypeaheadInput {
- const deserialized = Plain.deserialize(value);
- const range = deserialized.selection.merge({
- anchorOffset: anchorOffset || 1,
- });
- const valueWithSelection = deserialized.change().select(range).value;
- return {
- text,
- prefix: '',
- wrapperClasses: wrapperClasses || ['context-labels'],
- value: valueWithSelection,
- labelKey,
- };
- }
|