mirror of
https://github.com/grafana/grafana.git
synced 2025-02-20 11:48:34 -06:00
* Migrated loki syntax and labels logic to useLokiSyntax hook * Enable loki labels refresh after specified interval has passed * Enable periodic loki labels refresh when labels selector is opened * Fix prettier * Add react-hooks-testing-library and disable lib check on typecheck * Add tests for loki syntax/label hooks * Move tsc's skipLibCheck option to tsconfig for webpack to pick it up * Set log labels refresh marker variable when log labels fetch start * Fix prettier issues * Fix type on activeOption in useLokiLabel hook * Typo fixes and types in useLokiSyntax hook test fixes * Make sure effect's setState is not performed on unmounted component * Extract logic for checking if is component mounted to a separate hook
168 lines
5.9 KiB
TypeScript
168 lines
5.9 KiB
TypeScript
import Plain from 'slate-plain-serializer';
|
|
|
|
import LanguageProvider, { LABEL_REFRESH_INTERVAL } from './language_provider';
|
|
import { advanceTo, clear, advanceBy } from 'jest-date-mock';
|
|
import { beforeEach } from 'test/lib/common';
|
|
|
|
describe('Language completion provider', () => {
|
|
const datasource = {
|
|
metadataRequest: () => ({ data: { data: [] } }),
|
|
};
|
|
|
|
describe('empty query suggestions', () => {
|
|
it('returns no suggestions on emtpty 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 emtpty context when history was provided', () => {
|
|
const instance = new LanguageProvider(datasource);
|
|
const value = Plain.deserialize('');
|
|
const history = [
|
|
{
|
|
query: { refId: '1', expr: '{app="foo"}' },
|
|
},
|
|
];
|
|
const result = instance.provideCompletionItems({ text: '', prefix: '', value, wrapperClasses: [] }, { history });
|
|
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 value = Plain.deserialize('{} ()');
|
|
const range = value.selection.merge({
|
|
anchorOffset: 4,
|
|
});
|
|
const valueWithSelection = value.change().select(range).value;
|
|
const history = [
|
|
{
|
|
query: { refId: '1', expr: '{app="foo"}' },
|
|
},
|
|
];
|
|
const result = instance.provideCompletionItems(
|
|
{
|
|
text: '',
|
|
prefix: '',
|
|
value: valueWithSelection,
|
|
wrapperClasses: [],
|
|
},
|
|
{ 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 value = Plain.deserialize('{}');
|
|
const range = value.selection.merge({
|
|
anchorOffset: 1,
|
|
});
|
|
const valueWithSelection = value.change().select(range).value;
|
|
const result = instance.provideCompletionItems({
|
|
text: '',
|
|
prefix: '',
|
|
wrapperClasses: ['context-labels'],
|
|
value: valueWithSelection,
|
|
});
|
|
expect(result.context).toBe('context-labels');
|
|
expect(result.suggestions).toEqual([{ items: [{ label: 'job' }, { label: 'namespace' }], label: 'Labels' }]);
|
|
});
|
|
});
|
|
});
|
|
|
|
describe('Query imports', () => {
|
|
const datasource = {
|
|
metadataRequest: () => ({ data: { data: [] } }),
|
|
};
|
|
|
|
it('returns empty queries for unknown origin datasource', async () => {
|
|
const instance = new LanguageProvider(datasource);
|
|
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);
|
|
const result = await instance.importPrometheusQuery('foo');
|
|
expect(result).toEqual('');
|
|
});
|
|
|
|
it('returns empty query from selector query if label is not available', async () => {
|
|
const datasourceWithLabels = {
|
|
metadataRequest: url => (url === '/api/prom/label' ? { data: { data: ['other'] } } : { data: { data: [] } }),
|
|
};
|
|
const instance = new LanguageProvider(datasourceWithLabels);
|
|
const result = await instance.importPrometheusQuery('{foo="bar"}');
|
|
expect(result).toEqual('{}');
|
|
});
|
|
|
|
it('returns selector query from selector query with common labels', async () => {
|
|
const datasourceWithLabels = {
|
|
metadataRequest: url => (url === '/api/prom/label' ? { data: { data: ['foo'] } } : { data: { data: [] } }),
|
|
};
|
|
const instance = new LanguageProvider(datasourceWithLabels);
|
|
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 = {
|
|
metadataRequest: url => (url === '/api/prom/label' ? { data: { data: [] } } : { data: { data: [] } }),
|
|
};
|
|
const instance = new LanguageProvider(datasourceWithLabels);
|
|
const result = await instance.importPrometheusQuery('metric{foo="bar",baz="42"}');
|
|
expect(result).toEqual('{baz="42",foo="bar"}');
|
|
});
|
|
});
|
|
});
|
|
|
|
describe('Labels refresh', () => {
|
|
const datasource = {
|
|
metadataRequest: () => ({ data: { data: [] } }),
|
|
};
|
|
const instance = new LanguageProvider(datasource);
|
|
|
|
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();
|
|
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();
|
|
expect(instance.fetchLogLabels).toBeCalled();
|
|
});
|
|
});
|