The open and composable observability and data visualization platform. Visualize metrics, logs, and traces from multiple sources like Prometheus, Loki, Elasticsearch, InfluxDB, Postgres and many more.
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
 
 
 
 
 
 
grafana/packages/grafana-prometheus/src/components/usePromQueryFieldEffects.te...

204 lines
5.4 KiB

import { renderHook } from '@testing-library/react';
import { DataFrame, dateTime, TimeRange } from '@grafana/data';
import { PrometheusLanguageProviderInterface } from '../language_provider';
import { usePromQueryFieldEffects } from './usePromQueryFieldEffects';
type TestProps = {
languageProvider: PrometheusLanguageProviderInterface;
range: TimeRange | undefined;
series: DataFrame[];
};
describe('usePromQueryFieldEffects', () => {
const mockLanguageProvider = {
start: jest.fn().mockResolvedValue([]),
timeRange: {},
metrics: ['metric1'],
startTask: Promise.resolve(),
datasource: {},
lookupsDisabled: false,
syntax: jest.fn(),
hasLookupsDisabled: jest.fn(),
getBeginningCompletionItems: jest.fn(),
getLabelCompletionItems: jest.fn(),
getMetricCompletionItems: jest.fn(),
getTermCompletionItems: jest.fn(),
request: jest.fn(),
importQueries: jest.fn(),
labelFetchTs: 0,
getDefaultCacheHeaders: jest.fn(),
modifyQuery: jest.fn(),
} as unknown as PrometheusLanguageProviderInterface;
const mockRange: TimeRange = {
from: dateTime('2022-01-01T00:00:00Z'),
to: dateTime('2022-01-02T00:00:00Z'),
raw: {
from: 'now-1d',
to: 'now',
},
};
const mockNewRange: TimeRange = {
from: dateTime('2022-01-02T00:00:00Z'),
to: dateTime('2022-01-03T00:00:00Z'),
raw: {
from: 'now-1d',
to: 'now',
},
};
let refreshMetricsMock: jest.Mock;
let refreshHintMock: jest.Mock;
beforeEach(() => {
jest.clearAllMocks();
refreshMetricsMock = jest.fn().mockImplementation(() => Promise.resolve());
refreshHintMock = jest.fn();
});
it('should call refreshMetrics and refreshHint on initial render', async () => {
renderHook(() =>
usePromQueryFieldEffects(mockLanguageProvider, mockRange, [], refreshMetricsMock, refreshHintMock)
);
expect(refreshMetricsMock).toHaveBeenCalledTimes(1);
expect(refreshHintMock).toHaveBeenCalledTimes(2);
});
it('should call refreshMetrics when the time range changes', async () => {
const { rerender } = renderHook(
(props: TestProps) =>
usePromQueryFieldEffects(
props.languageProvider,
props.range,
props.series,
refreshMetricsMock,
refreshHintMock
),
{
initialProps: {
languageProvider: mockLanguageProvider,
range: mockRange,
series: [] as DataFrame[],
},
}
);
// Initial render already called refreshMetrics once
expect(refreshMetricsMock).toHaveBeenCalledTimes(1);
// Change the range
rerender({
languageProvider: mockLanguageProvider,
range: mockNewRange,
series: [] as DataFrame[],
});
expect(refreshMetricsMock).toHaveBeenCalledTimes(2);
});
it('should not call refreshMetrics when the time range is the same', () => {
const { rerender } = renderHook(
(props: TestProps) =>
usePromQueryFieldEffects(
props.languageProvider,
props.range,
props.series,
refreshMetricsMock,
refreshHintMock
),
{
initialProps: {
languageProvider: mockLanguageProvider,
range: mockRange,
series: [] as DataFrame[],
},
}
);
// Initial render already called refreshMetrics once
expect(refreshMetricsMock).toHaveBeenCalledTimes(1);
// Rerender with the same range
rerender({
languageProvider: mockLanguageProvider,
range: { ...mockRange }, // create a new object with the same values
series: [] as DataFrame[],
});
// Should still be called only once (from initial render)
expect(refreshMetricsMock).toHaveBeenCalledTimes(1);
});
it('should call refreshHint when series changes', () => {
const mockSeries = [{ name: 'new series', fields: [], length: 0 }] as DataFrame[];
const { rerender } = renderHook(
(props: TestProps) =>
usePromQueryFieldEffects(
props.languageProvider,
props.range,
props.series,
refreshMetricsMock,
refreshHintMock
),
{
initialProps: {
languageProvider: mockLanguageProvider,
range: mockRange,
series: [] as DataFrame[],
},
}
);
// Initial render already called refreshHint once
expect(refreshHintMock).toHaveBeenCalledTimes(2);
refreshHintMock.mockClear();
// Change the series
rerender({
languageProvider: mockLanguageProvider,
range: mockRange,
series: mockSeries,
});
expect(refreshHintMock).toHaveBeenCalledTimes(1);
});
it('should not call refreshHint when series is the same', () => {
const series = [] as DataFrame[];
const { rerender } = renderHook(
(props: TestProps) =>
usePromQueryFieldEffects(
props.languageProvider,
props.range,
props.series,
refreshMetricsMock,
refreshHintMock
),
{
initialProps: {
languageProvider: mockLanguageProvider,
range: mockRange,
series,
},
}
);
// Initial render already called refreshHint once
refreshHintMock.mockClear();
// Rerender with the same series
rerender({
languageProvider: mockLanguageProvider,
range: mockRange,
series, // same empty array
});
expect(refreshHintMock).not.toHaveBeenCalled();
});
});