|
@@ -1,22 +1,21 @@
|
|
|
|
|
+// @ts-ignore
|
|
|
import Plain from 'slate-plain-serializer';
|
|
import Plain from 'slate-plain-serializer';
|
|
|
-import { Editor as SlateEditor } from 'slate';
|
|
|
|
|
|
|
+
|
|
|
import LanguageProvider from '../language_provider';
|
|
import LanguageProvider from '../language_provider';
|
|
|
-import { PrometheusDatasource } from '../datasource';
|
|
|
|
|
-import { HistoryItem } from 'app/types';
|
|
|
|
|
-import { PromQuery } from '../types';
|
|
|
|
|
|
|
|
|
|
describe('Language completion provider', () => {
|
|
describe('Language completion provider', () => {
|
|
|
- const datasource: PrometheusDatasource = ({
|
|
|
|
|
|
|
+ const datasource = {
|
|
|
metadataRequest: () => ({ data: { data: [] as any[] } }),
|
|
metadataRequest: () => ({ data: { data: [] as any[] } }),
|
|
|
getTimeRange: () => ({ start: 0, end: 1 }),
|
|
getTimeRange: () => ({ start: 0, end: 1 }),
|
|
|
- } as any) as PrometheusDatasource;
|
|
|
|
|
|
|
+ };
|
|
|
|
|
|
|
|
describe('empty query suggestions', () => {
|
|
describe('empty query suggestions', () => {
|
|
|
- it('returns default suggestions on empty context', async () => {
|
|
|
|
|
|
|
+ it('returns default suggestions on emtpty context', () => {
|
|
|
const instance = new LanguageProvider(datasource);
|
|
const instance = new LanguageProvider(datasource);
|
|
|
const value = Plain.deserialize('');
|
|
const value = Plain.deserialize('');
|
|
|
- const result = await instance.provideCompletionItems({ text: '', prefix: '', value, wrapperClasses: [] });
|
|
|
|
|
|
|
+ const result = instance.provideCompletionItems({ text: '', prefix: '', value, wrapperClasses: [] });
|
|
|
expect(result.context).toBeUndefined();
|
|
expect(result.context).toBeUndefined();
|
|
|
|
|
+ expect(result.refresher).toBeUndefined();
|
|
|
expect(result.suggestions).toMatchObject([
|
|
expect(result.suggestions).toMatchObject([
|
|
|
{
|
|
{
|
|
|
label: 'Functions',
|
|
label: 'Functions',
|
|
@@ -24,11 +23,12 @@ describe('Language completion provider', () => {
|
|
|
]);
|
|
]);
|
|
|
});
|
|
});
|
|
|
|
|
|
|
|
- it('returns default suggestions with metrics on empty context when metrics were provided', async () => {
|
|
|
|
|
|
|
+ it('returns default suggestions with metrics on emtpty context when metrics were provided', () => {
|
|
|
const instance = new LanguageProvider(datasource, { metrics: ['foo', 'bar'] });
|
|
const instance = new LanguageProvider(datasource, { metrics: ['foo', 'bar'] });
|
|
|
const value = Plain.deserialize('');
|
|
const value = Plain.deserialize('');
|
|
|
- const result = await instance.provideCompletionItems({ text: '', prefix: '', value, wrapperClasses: [] });
|
|
|
|
|
|
|
+ const result = instance.provideCompletionItems({ text: '', prefix: '', value, wrapperClasses: [] });
|
|
|
expect(result.context).toBeUndefined();
|
|
expect(result.context).toBeUndefined();
|
|
|
|
|
+ expect(result.refresher).toBeUndefined();
|
|
|
expect(result.suggestions).toMatchObject([
|
|
expect(result.suggestions).toMatchObject([
|
|
|
{
|
|
{
|
|
|
label: 'Functions',
|
|
label: 'Functions',
|
|
@@ -39,21 +39,17 @@ describe('Language completion provider', () => {
|
|
|
]);
|
|
]);
|
|
|
});
|
|
});
|
|
|
|
|
|
|
|
- it('returns default suggestions with history on empty context when history was provided', async () => {
|
|
|
|
|
|
|
+ it('returns default suggestions with history on emtpty context when history was provided', () => {
|
|
|
const instance = new LanguageProvider(datasource);
|
|
const instance = new LanguageProvider(datasource);
|
|
|
const value = Plain.deserialize('');
|
|
const value = Plain.deserialize('');
|
|
|
- const history: Array<HistoryItem<PromQuery>> = [
|
|
|
|
|
|
|
+ const history = [
|
|
|
{
|
|
{
|
|
|
- ts: 0,
|
|
|
|
|
query: { refId: '1', expr: 'metric' },
|
|
query: { refId: '1', expr: 'metric' },
|
|
|
},
|
|
},
|
|
|
];
|
|
];
|
|
|
- const result = await instance.provideCompletionItems(
|
|
|
|
|
- { text: '', prefix: '', value, wrapperClasses: [] },
|
|
|
|
|
- { history }
|
|
|
|
|
- );
|
|
|
|
|
|
|
+ const result = instance.provideCompletionItems({ text: '', prefix: '', value, wrapperClasses: [] }, { history });
|
|
|
expect(result.context).toBeUndefined();
|
|
expect(result.context).toBeUndefined();
|
|
|
-
|
|
|
|
|
|
|
+ expect(result.refresher).toBeUndefined();
|
|
|
expect(result.suggestions).toMatchObject([
|
|
expect(result.suggestions).toMatchObject([
|
|
|
{
|
|
{
|
|
|
label: 'History',
|
|
label: 'History',
|
|
@@ -71,16 +67,17 @@ describe('Language completion provider', () => {
|
|
|
});
|
|
});
|
|
|
|
|
|
|
|
describe('range suggestions', () => {
|
|
describe('range suggestions', () => {
|
|
|
- it('returns range suggestions in range context', async () => {
|
|
|
|
|
|
|
+ it('returns range suggestions in range context', () => {
|
|
|
const instance = new LanguageProvider(datasource);
|
|
const instance = new LanguageProvider(datasource);
|
|
|
const value = Plain.deserialize('1');
|
|
const value = Plain.deserialize('1');
|
|
|
- const result = await instance.provideCompletionItems({
|
|
|
|
|
|
|
+ const result = instance.provideCompletionItems({
|
|
|
text: '1',
|
|
text: '1',
|
|
|
prefix: '1',
|
|
prefix: '1',
|
|
|
value,
|
|
value,
|
|
|
wrapperClasses: ['context-range'],
|
|
wrapperClasses: ['context-range'],
|
|
|
});
|
|
});
|
|
|
expect(result.context).toBe('context-range');
|
|
expect(result.context).toBe('context-range');
|
|
|
|
|
+ expect(result.refresher).toBeUndefined();
|
|
|
expect(result.suggestions).toMatchObject([
|
|
expect(result.suggestions).toMatchObject([
|
|
|
{
|
|
{
|
|
|
items: [
|
|
items: [
|
|
@@ -99,12 +96,12 @@ describe('Language completion provider', () => {
|
|
|
});
|
|
});
|
|
|
|
|
|
|
|
describe('metric suggestions', () => {
|
|
describe('metric suggestions', () => {
|
|
|
- it('returns metrics and function suggestions in an unknown context', async () => {
|
|
|
|
|
|
|
+ it('returns metrics and function suggestions in an unknown context', () => {
|
|
|
const instance = new LanguageProvider(datasource, { metrics: ['foo', 'bar'] });
|
|
const instance = new LanguageProvider(datasource, { metrics: ['foo', 'bar'] });
|
|
|
- let value = Plain.deserialize('a');
|
|
|
|
|
- value = value.setSelection({ anchor: { offset: 1 }, focus: { offset: 1 } });
|
|
|
|
|
- const result = await instance.provideCompletionItems({ text: 'a', prefix: 'a', value, wrapperClasses: [] });
|
|
|
|
|
|
|
+ const value = Plain.deserialize('a');
|
|
|
|
|
+ const result = instance.provideCompletionItems({ text: 'a', prefix: 'a', value, wrapperClasses: [] });
|
|
|
expect(result.context).toBeUndefined();
|
|
expect(result.context).toBeUndefined();
|
|
|
|
|
+ expect(result.refresher).toBeUndefined();
|
|
|
expect(result.suggestions).toMatchObject([
|
|
expect(result.suggestions).toMatchObject([
|
|
|
{
|
|
{
|
|
|
label: 'Functions',
|
|
label: 'Functions',
|
|
@@ -115,11 +112,12 @@ describe('Language completion provider', () => {
|
|
|
]);
|
|
]);
|
|
|
});
|
|
});
|
|
|
|
|
|
|
|
- it('returns metrics and function suggestions after a binary operator', async () => {
|
|
|
|
|
|
|
+ it('returns metrics and function suggestions after a binary operator', () => {
|
|
|
const instance = new LanguageProvider(datasource, { metrics: ['foo', 'bar'] });
|
|
const instance = new LanguageProvider(datasource, { metrics: ['foo', 'bar'] });
|
|
|
const value = Plain.deserialize('*');
|
|
const value = Plain.deserialize('*');
|
|
|
- const result = await instance.provideCompletionItems({ text: '*', prefix: '', value, wrapperClasses: [] });
|
|
|
|
|
|
|
+ const result = instance.provideCompletionItems({ text: '*', prefix: '', value, wrapperClasses: [] });
|
|
|
expect(result.context).toBeUndefined();
|
|
expect(result.context).toBeUndefined();
|
|
|
|
|
+ expect(result.refresher).toBeUndefined();
|
|
|
expect(result.suggestions).toMatchObject([
|
|
expect(result.suggestions).toMatchObject([
|
|
|
{
|
|
{
|
|
|
label: 'Functions',
|
|
label: 'Functions',
|
|
@@ -130,30 +128,34 @@ describe('Language completion provider', () => {
|
|
|
]);
|
|
]);
|
|
|
});
|
|
});
|
|
|
|
|
|
|
|
- it('returns no suggestions at the beginning of a non-empty function', async () => {
|
|
|
|
|
|
|
+ it('returns no suggestions at the beginning of a non-empty function', () => {
|
|
|
const instance = new LanguageProvider(datasource, { metrics: ['foo', 'bar'] });
|
|
const instance = new LanguageProvider(datasource, { metrics: ['foo', 'bar'] });
|
|
|
const value = Plain.deserialize('sum(up)');
|
|
const value = Plain.deserialize('sum(up)');
|
|
|
- const ed = new SlateEditor({ value });
|
|
|
|
|
-
|
|
|
|
|
- const valueWithSelection = ed.moveForward(4).value;
|
|
|
|
|
- const result = await instance.provideCompletionItems({
|
|
|
|
|
|
|
+ const range = value.selection.merge({
|
|
|
|
|
+ anchorOffset: 4,
|
|
|
|
|
+ });
|
|
|
|
|
+ const valueWithSelection = value.change().select(range).value;
|
|
|
|
|
+ const result = instance.provideCompletionItems({
|
|
|
text: '',
|
|
text: '',
|
|
|
prefix: '',
|
|
prefix: '',
|
|
|
value: valueWithSelection,
|
|
value: valueWithSelection,
|
|
|
wrapperClasses: [],
|
|
wrapperClasses: [],
|
|
|
});
|
|
});
|
|
|
expect(result.context).toBeUndefined();
|
|
expect(result.context).toBeUndefined();
|
|
|
|
|
+ expect(result.refresher).toBeUndefined();
|
|
|
expect(result.suggestions.length).toEqual(0);
|
|
expect(result.suggestions.length).toEqual(0);
|
|
|
});
|
|
});
|
|
|
});
|
|
});
|
|
|
|
|
|
|
|
describe('label suggestions', () => {
|
|
describe('label suggestions', () => {
|
|
|
- it('returns default label suggestions on label context and no metric', async () => {
|
|
|
|
|
|
|
+ it('returns default label suggestions on label context and no metric', () => {
|
|
|
const instance = new LanguageProvider(datasource);
|
|
const instance = new LanguageProvider(datasource);
|
|
|
const value = Plain.deserialize('{}');
|
|
const value = Plain.deserialize('{}');
|
|
|
- const ed = new SlateEditor({ value });
|
|
|
|
|
- const valueWithSelection = ed.moveForward(1).value;
|
|
|
|
|
- const result = await instance.provideCompletionItems({
|
|
|
|
|
|
|
+ const range = value.selection.merge({
|
|
|
|
|
+ anchorOffset: 1,
|
|
|
|
|
+ });
|
|
|
|
|
+ const valueWithSelection = value.change().select(range).value;
|
|
|
|
|
+ const result = instance.provideCompletionItems({
|
|
|
text: '',
|
|
text: '',
|
|
|
prefix: '',
|
|
prefix: '',
|
|
|
wrapperClasses: ['context-labels'],
|
|
wrapperClasses: ['context-labels'],
|
|
@@ -163,16 +165,14 @@ describe('Language completion provider', () => {
|
|
|
expect(result.suggestions).toEqual([{ items: [{ label: 'job' }, { label: 'instance' }], label: 'Labels' }]);
|
|
expect(result.suggestions).toEqual([{ items: [{ label: 'job' }, { label: 'instance' }], label: 'Labels' }]);
|
|
|
});
|
|
});
|
|
|
|
|
|
|
|
- it('returns label suggestions on label context and metric', async () => {
|
|
|
|
|
- const datasources: PrometheusDatasource = ({
|
|
|
|
|
- metadataRequest: () => ({ data: { data: [{ __name__: 'metric', bar: 'bazinga' }] as any[] } }),
|
|
|
|
|
- getTimeRange: () => ({ start: 0, end: 1 }),
|
|
|
|
|
- } as any) as PrometheusDatasource;
|
|
|
|
|
- const instance = new LanguageProvider(datasources, { labelKeys: { '{__name__="metric"}': ['bar'] } });
|
|
|
|
|
|
|
+ it('returns label suggestions on label context and metric', () => {
|
|
|
|
|
+ const instance = new LanguageProvider(datasource, { labelKeys: { '{__name__="metric"}': ['bar'] } });
|
|
|
const value = Plain.deserialize('metric{}');
|
|
const value = Plain.deserialize('metric{}');
|
|
|
- const ed = new SlateEditor({ value });
|
|
|
|
|
- const valueWithSelection = ed.moveForward(7).value;
|
|
|
|
|
- const result = await instance.provideCompletionItems({
|
|
|
|
|
|
|
+ const range = value.selection.merge({
|
|
|
|
|
+ anchorOffset: 7,
|
|
|
|
|
+ });
|
|
|
|
|
+ const valueWithSelection = value.change().select(range).value;
|
|
|
|
|
+ const result = instance.provideCompletionItems({
|
|
|
text: '',
|
|
text: '',
|
|
|
prefix: '',
|
|
prefix: '',
|
|
|
wrapperClasses: ['context-labels'],
|
|
wrapperClasses: ['context-labels'],
|
|
@@ -182,32 +182,16 @@ describe('Language completion provider', () => {
|
|
|
expect(result.suggestions).toEqual([{ items: [{ label: 'bar' }], label: 'Labels' }]);
|
|
expect(result.suggestions).toEqual([{ items: [{ label: 'bar' }], label: 'Labels' }]);
|
|
|
});
|
|
});
|
|
|
|
|
|
|
|
- it('returns label suggestions on label context but leaves out labels that already exist', async () => {
|
|
|
|
|
- const datasources: PrometheusDatasource = ({
|
|
|
|
|
- metadataRequest: () => ({
|
|
|
|
|
- data: {
|
|
|
|
|
- data: [
|
|
|
|
|
- {
|
|
|
|
|
- __name__: 'metric',
|
|
|
|
|
- bar: 'asdasd',
|
|
|
|
|
- job1: 'dsadsads',
|
|
|
|
|
- job2: 'fsfsdfds',
|
|
|
|
|
- job3: 'dsadsad',
|
|
|
|
|
- },
|
|
|
|
|
- ],
|
|
|
|
|
- },
|
|
|
|
|
- }),
|
|
|
|
|
- getTimeRange: () => ({ start: 0, end: 1 }),
|
|
|
|
|
- } as any) as PrometheusDatasource;
|
|
|
|
|
- const instance = new LanguageProvider(datasources, {
|
|
|
|
|
- labelKeys: {
|
|
|
|
|
- '{job1="foo",job2!="foo",job3=~"foo",__name__="metric"}': ['bar', 'job1', 'job2', 'job3', '__name__'],
|
|
|
|
|
- },
|
|
|
|
|
|
|
+ it('returns label suggestions on label context but leaves out labels that already exist', () => {
|
|
|
|
|
+ const instance = new LanguageProvider(datasource, {
|
|
|
|
|
+ labelKeys: { '{job1="foo",job2!="foo",job3=~"foo"}': ['bar', 'job1', 'job2', 'job3'] },
|
|
|
});
|
|
});
|
|
|
- const value = Plain.deserialize('{job1="foo",job2!="foo",job3=~"foo",__name__="metric",}');
|
|
|
|
|
- const ed = new SlateEditor({ value });
|
|
|
|
|
- const valueWithSelection = ed.moveForward(54).value;
|
|
|
|
|
- const result = await instance.provideCompletionItems({
|
|
|
|
|
|
|
+ const value = Plain.deserialize('{job1="foo",job2!="foo",job3=~"foo",}');
|
|
|
|
|
+ const range = value.selection.merge({
|
|
|
|
|
+ anchorOffset: 36,
|
|
|
|
|
+ });
|
|
|
|
|
+ const valueWithSelection = value.change().select(range).value;
|
|
|
|
|
+ const result = instance.provideCompletionItems({
|
|
|
text: '',
|
|
text: '',
|
|
|
prefix: '',
|
|
prefix: '',
|
|
|
wrapperClasses: ['context-labels'],
|
|
wrapperClasses: ['context-labels'],
|
|
@@ -217,15 +201,15 @@ describe('Language completion provider', () => {
|
|
|
expect(result.suggestions).toEqual([{ items: [{ label: 'bar' }], label: 'Labels' }]);
|
|
expect(result.suggestions).toEqual([{ items: [{ label: 'bar' }], label: 'Labels' }]);
|
|
|
});
|
|
});
|
|
|
|
|
|
|
|
- it('returns label value suggestions inside a label value context after a negated matching operator', async () => {
|
|
|
|
|
|
|
+ it('returns label value suggestions inside a label value context after a negated matching operator', () => {
|
|
|
const instance = new LanguageProvider(datasource, {
|
|
const instance = new LanguageProvider(datasource, {
|
|
|
labelKeys: { '{}': ['label'] },
|
|
labelKeys: { '{}': ['label'] },
|
|
|
labelValues: { '{}': { label: ['a', 'b', 'c'] } },
|
|
labelValues: { '{}': { label: ['a', 'b', 'c'] } },
|
|
|
});
|
|
});
|
|
|
const value = Plain.deserialize('{label!=}');
|
|
const value = Plain.deserialize('{label!=}');
|
|
|
- const ed = new SlateEditor({ value });
|
|
|
|
|
- const valueWithSelection = ed.moveForward(8).value;
|
|
|
|
|
- const result = await instance.provideCompletionItems({
|
|
|
|
|
|
|
+ const range = value.selection.merge({ anchorOffset: 8 });
|
|
|
|
|
+ const valueWithSelection = value.change().select(range).value;
|
|
|
|
|
+ const result = instance.provideCompletionItems({
|
|
|
text: '!=',
|
|
text: '!=',
|
|
|
prefix: '',
|
|
prefix: '',
|
|
|
wrapperClasses: ['context-labels'],
|
|
wrapperClasses: ['context-labels'],
|
|
@@ -241,30 +225,35 @@ describe('Language completion provider', () => {
|
|
|
]);
|
|
]);
|
|
|
});
|
|
});
|
|
|
|
|
|
|
|
- it('returns a refresher on label context and unavailable metric', async () => {
|
|
|
|
|
|
|
+ it('returns a refresher on label context and unavailable metric', () => {
|
|
|
const instance = new LanguageProvider(datasource, { labelKeys: { '{__name__="foo"}': ['bar'] } });
|
|
const instance = new LanguageProvider(datasource, { labelKeys: { '{__name__="foo"}': ['bar'] } });
|
|
|
const value = Plain.deserialize('metric{}');
|
|
const value = Plain.deserialize('metric{}');
|
|
|
- const ed = new SlateEditor({ value });
|
|
|
|
|
- const valueWithSelection = ed.moveForward(7).value;
|
|
|
|
|
- const result = await instance.provideCompletionItems({
|
|
|
|
|
|
|
+ const range = value.selection.merge({
|
|
|
|
|
+ anchorOffset: 7,
|
|
|
|
|
+ });
|
|
|
|
|
+ const valueWithSelection = value.change().select(range).value;
|
|
|
|
|
+ const result = instance.provideCompletionItems({
|
|
|
text: '',
|
|
text: '',
|
|
|
prefix: '',
|
|
prefix: '',
|
|
|
wrapperClasses: ['context-labels'],
|
|
wrapperClasses: ['context-labels'],
|
|
|
value: valueWithSelection,
|
|
value: valueWithSelection,
|
|
|
});
|
|
});
|
|
|
expect(result.context).toBeUndefined();
|
|
expect(result.context).toBeUndefined();
|
|
|
|
|
+ expect(result.refresher).toBeInstanceOf(Promise);
|
|
|
expect(result.suggestions).toEqual([]);
|
|
expect(result.suggestions).toEqual([]);
|
|
|
});
|
|
});
|
|
|
|
|
|
|
|
- it('returns label values on label context when given a metric and a label key', async () => {
|
|
|
|
|
|
|
+ it('returns label values on label context when given a metric and a label key', () => {
|
|
|
const instance = new LanguageProvider(datasource, {
|
|
const instance = new LanguageProvider(datasource, {
|
|
|
labelKeys: { '{__name__="metric"}': ['bar'] },
|
|
labelKeys: { '{__name__="metric"}': ['bar'] },
|
|
|
labelValues: { '{__name__="metric"}': { bar: ['baz'] } },
|
|
labelValues: { '{__name__="metric"}': { bar: ['baz'] } },
|
|
|
});
|
|
});
|
|
|
const value = Plain.deserialize('metric{bar=ba}');
|
|
const value = Plain.deserialize('metric{bar=ba}');
|
|
|
- const ed = new SlateEditor({ value });
|
|
|
|
|
- const valueWithSelection = ed.moveForward(13).value;
|
|
|
|
|
- const result = await instance.provideCompletionItems({
|
|
|
|
|
|
|
+ const range = value.selection.merge({
|
|
|
|
|
+ anchorOffset: 13,
|
|
|
|
|
+ });
|
|
|
|
|
+ const valueWithSelection = value.change().select(range).value;
|
|
|
|
|
+ const result = instance.provideCompletionItems({
|
|
|
text: '=ba',
|
|
text: '=ba',
|
|
|
prefix: 'ba',
|
|
prefix: 'ba',
|
|
|
wrapperClasses: ['context-labels'],
|
|
wrapperClasses: ['context-labels'],
|
|
@@ -275,12 +264,14 @@ describe('Language completion provider', () => {
|
|
|
expect(result.suggestions).toEqual([{ items: [{ label: 'baz' }], label: 'Label values for "bar"' }]);
|
|
expect(result.suggestions).toEqual([{ items: [{ label: 'baz' }], label: 'Label values for "bar"' }]);
|
|
|
});
|
|
});
|
|
|
|
|
|
|
|
- it('returns label suggestions on aggregation context and metric w/ selector', async () => {
|
|
|
|
|
|
|
+ it('returns label suggestions on aggregation context and metric w/ selector', () => {
|
|
|
const instance = new LanguageProvider(datasource, { labelKeys: { '{__name__="metric",foo="xx"}': ['bar'] } });
|
|
const instance = new LanguageProvider(datasource, { labelKeys: { '{__name__="metric",foo="xx"}': ['bar'] } });
|
|
|
const value = Plain.deserialize('sum(metric{foo="xx"}) by ()');
|
|
const value = Plain.deserialize('sum(metric{foo="xx"}) by ()');
|
|
|
- const ed = new SlateEditor({ value });
|
|
|
|
|
- const valueWithSelection = ed.moveForward(26).value;
|
|
|
|
|
- const result = await instance.provideCompletionItems({
|
|
|
|
|
|
|
+ const range = value.selection.merge({
|
|
|
|
|
+ anchorOffset: 26,
|
|
|
|
|
+ });
|
|
|
|
|
+ const valueWithSelection = value.change().select(range).value;
|
|
|
|
|
+ const result = instance.provideCompletionItems({
|
|
|
text: '',
|
|
text: '',
|
|
|
prefix: '',
|
|
prefix: '',
|
|
|
wrapperClasses: ['context-aggregation'],
|
|
wrapperClasses: ['context-aggregation'],
|
|
@@ -290,12 +281,14 @@ describe('Language completion provider', () => {
|
|
|
expect(result.suggestions).toEqual([{ items: [{ label: 'bar' }], label: 'Labels' }]);
|
|
expect(result.suggestions).toEqual([{ items: [{ label: 'bar' }], label: 'Labels' }]);
|
|
|
});
|
|
});
|
|
|
|
|
|
|
|
- it('returns label suggestions on aggregation context and metric w/o selector', async () => {
|
|
|
|
|
|
|
+ it('returns label suggestions on aggregation context and metric w/o selector', () => {
|
|
|
const instance = new LanguageProvider(datasource, { labelKeys: { '{__name__="metric"}': ['bar'] } });
|
|
const instance = new LanguageProvider(datasource, { labelKeys: { '{__name__="metric"}': ['bar'] } });
|
|
|
const value = Plain.deserialize('sum(metric) by ()');
|
|
const value = Plain.deserialize('sum(metric) by ()');
|
|
|
- const ed = new SlateEditor({ value });
|
|
|
|
|
- const valueWithSelection = ed.moveForward(16).value;
|
|
|
|
|
- const result = await instance.provideCompletionItems({
|
|
|
|
|
|
|
+ const range = value.selection.merge({
|
|
|
|
|
+ anchorOffset: 16,
|
|
|
|
|
+ });
|
|
|
|
|
+ const valueWithSelection = value.change().select(range).value;
|
|
|
|
|
+ const result = instance.provideCompletionItems({
|
|
|
text: '',
|
|
text: '',
|
|
|
prefix: '',
|
|
prefix: '',
|
|
|
wrapperClasses: ['context-aggregation'],
|
|
wrapperClasses: ['context-aggregation'],
|
|
@@ -305,16 +298,15 @@ describe('Language completion provider', () => {
|
|
|
expect(result.suggestions).toEqual([{ items: [{ label: 'bar' }], label: 'Labels' }]);
|
|
expect(result.suggestions).toEqual([{ items: [{ label: 'bar' }], label: 'Labels' }]);
|
|
|
});
|
|
});
|
|
|
|
|
|
|
|
- it('returns label suggestions inside a multi-line aggregation context', async () => {
|
|
|
|
|
|
|
+ it('returns label suggestions inside a multi-line aggregation context', () => {
|
|
|
const instance = new LanguageProvider(datasource, {
|
|
const instance = new LanguageProvider(datasource, {
|
|
|
labelKeys: { '{__name__="metric"}': ['label1', 'label2', 'label3'] },
|
|
labelKeys: { '{__name__="metric"}': ['label1', 'label2', 'label3'] },
|
|
|
});
|
|
});
|
|
|
const value = Plain.deserialize('sum(\nmetric\n)\nby ()');
|
|
const value = Plain.deserialize('sum(\nmetric\n)\nby ()');
|
|
|
- const aggregationTextBlock = value.document.getBlocks().get(3);
|
|
|
|
|
- const ed = new SlateEditor({ value });
|
|
|
|
|
- ed.moveToStartOfNode(aggregationTextBlock);
|
|
|
|
|
- const valueWithSelection = ed.moveForward(4).value;
|
|
|
|
|
- const result = await instance.provideCompletionItems({
|
|
|
|
|
|
|
+ const aggregationTextBlock = value.document.getBlocksAsArray()[3];
|
|
|
|
|
+ const range = value.selection.moveToStartOf(aggregationTextBlock).merge({ anchorOffset: 4 });
|
|
|
|
|
+ const valueWithSelection = value.change().select(range).value;
|
|
|
|
|
+ const result = instance.provideCompletionItems({
|
|
|
text: '',
|
|
text: '',
|
|
|
prefix: '',
|
|
prefix: '',
|
|
|
wrapperClasses: ['context-aggregation'],
|
|
wrapperClasses: ['context-aggregation'],
|
|
@@ -329,14 +321,16 @@ describe('Language completion provider', () => {
|
|
|
]);
|
|
]);
|
|
|
});
|
|
});
|
|
|
|
|
|
|
|
- it('returns label suggestions inside an aggregation context with a range vector', async () => {
|
|
|
|
|
|
|
+ it('returns label suggestions inside an aggregation context with a range vector', () => {
|
|
|
const instance = new LanguageProvider(datasource, {
|
|
const instance = new LanguageProvider(datasource, {
|
|
|
labelKeys: { '{__name__="metric"}': ['label1', 'label2', 'label3'] },
|
|
labelKeys: { '{__name__="metric"}': ['label1', 'label2', 'label3'] },
|
|
|
});
|
|
});
|
|
|
const value = Plain.deserialize('sum(rate(metric[1h])) by ()');
|
|
const value = Plain.deserialize('sum(rate(metric[1h])) by ()');
|
|
|
- const ed = new SlateEditor({ value });
|
|
|
|
|
- const valueWithSelection = ed.moveForward(26).value;
|
|
|
|
|
- const result = await instance.provideCompletionItems({
|
|
|
|
|
|
|
+ const range = value.selection.merge({
|
|
|
|
|
+ anchorOffset: 26,
|
|
|
|
|
+ });
|
|
|
|
|
+ const valueWithSelection = value.change().select(range).value;
|
|
|
|
|
+ const result = instance.provideCompletionItems({
|
|
|
text: '',
|
|
text: '',
|
|
|
prefix: '',
|
|
prefix: '',
|
|
|
wrapperClasses: ['context-aggregation'],
|
|
wrapperClasses: ['context-aggregation'],
|
|
@@ -351,14 +345,16 @@ describe('Language completion provider', () => {
|
|
|
]);
|
|
]);
|
|
|
});
|
|
});
|
|
|
|
|
|
|
|
- it('returns label suggestions inside an aggregation context with a range vector and label', async () => {
|
|
|
|
|
|
|
+ it('returns label suggestions inside an aggregation context with a range vector and label', () => {
|
|
|
const instance = new LanguageProvider(datasource, {
|
|
const instance = new LanguageProvider(datasource, {
|
|
|
labelKeys: { '{__name__="metric",label1="value"}': ['label1', 'label2', 'label3'] },
|
|
labelKeys: { '{__name__="metric",label1="value"}': ['label1', 'label2', 'label3'] },
|
|
|
});
|
|
});
|
|
|
const value = Plain.deserialize('sum(rate(metric{label1="value"}[1h])) by ()');
|
|
const value = Plain.deserialize('sum(rate(metric{label1="value"}[1h])) by ()');
|
|
|
- const ed = new SlateEditor({ value });
|
|
|
|
|
- const valueWithSelection = ed.moveForward(42).value;
|
|
|
|
|
- const result = await instance.provideCompletionItems({
|
|
|
|
|
|
|
+ const range = value.selection.merge({
|
|
|
|
|
+ anchorOffset: 42,
|
|
|
|
|
+ });
|
|
|
|
|
+ const valueWithSelection = value.change().select(range).value;
|
|
|
|
|
+ const result = instance.provideCompletionItems({
|
|
|
text: '',
|
|
text: '',
|
|
|
prefix: '',
|
|
prefix: '',
|
|
|
wrapperClasses: ['context-aggregation'],
|
|
wrapperClasses: ['context-aggregation'],
|
|
@@ -373,14 +369,16 @@ describe('Language completion provider', () => {
|
|
|
]);
|
|
]);
|
|
|
});
|
|
});
|
|
|
|
|
|
|
|
- it('returns no suggestions inside an unclear aggregation context using alternate syntax', async () => {
|
|
|
|
|
|
|
+ it('returns no suggestions inside an unclear aggregation context using alternate syntax', () => {
|
|
|
const instance = new LanguageProvider(datasource, {
|
|
const instance = new LanguageProvider(datasource, {
|
|
|
labelKeys: { '{__name__="metric"}': ['label1', 'label2', 'label3'] },
|
|
labelKeys: { '{__name__="metric"}': ['label1', 'label2', 'label3'] },
|
|
|
});
|
|
});
|
|
|
const value = Plain.deserialize('sum by ()');
|
|
const value = Plain.deserialize('sum by ()');
|
|
|
- const ed = new SlateEditor({ value });
|
|
|
|
|
- const valueWithSelection = ed.moveForward(8).value;
|
|
|
|
|
- const result = await instance.provideCompletionItems({
|
|
|
|
|
|
|
+ const range = value.selection.merge({
|
|
|
|
|
+ anchorOffset: 8,
|
|
|
|
|
+ });
|
|
|
|
|
+ const valueWithSelection = value.change().select(range).value;
|
|
|
|
|
+ const result = instance.provideCompletionItems({
|
|
|
text: '',
|
|
text: '',
|
|
|
prefix: '',
|
|
prefix: '',
|
|
|
wrapperClasses: ['context-aggregation'],
|
|
wrapperClasses: ['context-aggregation'],
|
|
@@ -390,14 +388,16 @@ describe('Language completion provider', () => {
|
|
|
expect(result.suggestions).toEqual([]);
|
|
expect(result.suggestions).toEqual([]);
|
|
|
});
|
|
});
|
|
|
|
|
|
|
|
- it('returns label suggestions inside an aggregation context using alternate syntax', async () => {
|
|
|
|
|
|
|
+ it('returns label suggestions inside an aggregation context using alternate syntax', () => {
|
|
|
const instance = new LanguageProvider(datasource, {
|
|
const instance = new LanguageProvider(datasource, {
|
|
|
labelKeys: { '{__name__="metric"}': ['label1', 'label2', 'label3'] },
|
|
labelKeys: { '{__name__="metric"}': ['label1', 'label2', 'label3'] },
|
|
|
});
|
|
});
|
|
|
const value = Plain.deserialize('sum by () (metric)');
|
|
const value = Plain.deserialize('sum by () (metric)');
|
|
|
- const ed = new SlateEditor({ value });
|
|
|
|
|
- const valueWithSelection = ed.moveForward(8).value;
|
|
|
|
|
- const result = await instance.provideCompletionItems({
|
|
|
|
|
|
|
+ const range = value.selection.merge({
|
|
|
|
|
+ anchorOffset: 8,
|
|
|
|
|
+ });
|
|
|
|
|
+ const valueWithSelection = value.change().select(range).value;
|
|
|
|
|
+ const result = instance.provideCompletionItems({
|
|
|
text: '',
|
|
text: '',
|
|
|
prefix: '',
|
|
prefix: '',
|
|
|
wrapperClasses: ['context-aggregation'],
|
|
wrapperClasses: ['context-aggregation'],
|