Explore: Refactor & centralize URL/state sync (#66286)

Co-authored-by: Piotr Jamróz <pm.jamroz@gmail.com>
This commit is contained in:
Giordano Ricci
2023-06-06 15:31:39 +01:00
committed by GitHub
parent 6900336f09
commit 067bbcbe56
42 changed files with 1360 additions and 1441 deletions

View File

@@ -1675,10 +1675,7 @@ exports[`better eslint`] = {
[0, 0, 0, "Unexpected any. Specify a different type.", "1"], [0, 0, 0, "Unexpected any. Specify a different type.", "1"],
[0, 0, 0, "Unexpected any. Specify a different type.", "2"], [0, 0, 0, "Unexpected any. Specify a different type.", "2"],
[0, 0, 0, "Unexpected any. Specify a different type.", "3"], [0, 0, 0, "Unexpected any. Specify a different type.", "3"],
[0, 0, 0, "Unexpected any. Specify a different type.", "4"], [0, 0, 0, "Unexpected any. Specify a different type.", "4"]
[0, 0, 0, "Unexpected any. Specify a different type.", "5"],
[0, 0, 0, "Unexpected any. Specify a different type.", "6"],
[0, 0, 0, "Unexpected any. Specify a different type.", "7"]
], ],
"public/app/core/utils/fetch.ts:5381": [ "public/app/core/utils/fetch.ts:5381": [
[0, 0, 0, "Do not use any type assertions.", "0"], [0, 0, 0, "Do not use any type assertions.", "0"],
@@ -2560,6 +2557,9 @@ exports[`better eslint`] = {
"public/app/features/explore/ElapsedTime.tsx:5381": [ "public/app/features/explore/ElapsedTime.tsx:5381": [
[0, 0, 0, "Unexpected any. Specify a different type.", "0"] [0, 0, 0, "Unexpected any. Specify a different type.", "0"]
], ],
"public/app/features/explore/ExplorePage.tsx:5381": [
[0, 0, 0, "Do not use any type assertions.", "0"]
],
"public/app/features/explore/ExploreQueryInspector.tsx:5381": [ "public/app/features/explore/ExploreQueryInspector.tsx:5381": [
[0, 0, 0, "Do not use any type assertions.", "0"] [0, 0, 0, "Do not use any type assertions.", "0"]
], ],
@@ -2620,14 +2620,14 @@ exports[`better eslint`] = {
[0, 0, 0, "Do not use any type assertions.", "0"], [0, 0, 0, "Do not use any type assertions.", "0"],
[0, 0, 0, "Do not use any type assertions.", "1"] [0, 0, 0, "Do not use any type assertions.", "1"]
], ],
"public/app/features/explore/hooks/useStateSync.ts:5381": [
[0, 0, 0, "Do not use any type assertions.", "0"],
[0, 0, 0, "Do not use any type assertions.", "1"],
[0, 0, 0, "Do not use any type assertions.", "2"]
],
"public/app/features/explore/spec/helper/setup.tsx:5381": [ "public/app/features/explore/spec/helper/setup.tsx:5381": [
[0, 0, 0, "Do not use any type assertions.", "0"], [0, 0, 0, "Do not use any type assertions.", "0"],
[0, 0, 0, "Unexpected any. Specify a different type.", "1"], [0, 0, 0, "Unexpected any. Specify a different type.", "1"]
[0, 0, 0, "Do not use any type assertions.", "2"],
[0, 0, 0, "Unexpected any. Specify a different type.", "3"],
[0, 0, 0, "Unexpected any. Specify a different type.", "4"],
[0, 0, 0, "Do not use any type assertions.", "5"],
[0, 0, 0, "Unexpected any. Specify a different type.", "6"]
], ],
"public/app/features/explore/spec/interpolation.test.tsx:5381": [ "public/app/features/explore/spec/interpolation.test.tsx:5381": [
[0, 0, 0, "Unexpected any. Specify a different type.", "0"] [0, 0, 0, "Unexpected any. Specify a different type.", "0"]
@@ -2635,19 +2635,11 @@ exports[`better eslint`] = {
"public/app/features/explore/spec/queryHistory.test.tsx:5381": [ "public/app/features/explore/spec/queryHistory.test.tsx:5381": [
[0, 0, 0, "Unexpected any. Specify a different type.", "0"] [0, 0, 0, "Unexpected any. Specify a different type.", "0"]
], ],
"public/app/features/explore/state/explorePane.test.ts:5381": [
[0, 0, 0, "Unexpected any. Specify a different type.", "0"],
[0, 0, 0, "Unexpected any. Specify a different type.", "1"],
[0, 0, 0, "Unexpected any. Specify a different type.", "2"],
[0, 0, 0, "Unexpected any. Specify a different type.", "3"],
[0, 0, 0, "Unexpected any. Specify a different type.", "4"]
],
"public/app/features/explore/state/history.ts:5381": [ "public/app/features/explore/state/history.ts:5381": [
[0, 0, 0, "Do not use any type assertions.", "0"] [0, 0, 0, "Do not use any type assertions.", "0"]
], ],
"public/app/features/explore/state/main.ts:5381": [ "public/app/features/explore/state/main.ts:5381": [
[0, 0, 0, "Do not use any type assertions.", "0"], [0, 0, 0, "Do not use any type assertions.", "0"]
[0, 0, 0, "Do not use any type assertions.", "1"]
], ],
"public/app/features/explore/state/query.ts:5381": [ "public/app/features/explore/state/query.ts:5381": [
[0, 0, 0, "Do not use any type assertions.", "0"] [0, 0, 0, "Do not use any type assertions.", "0"]
@@ -5773,7 +5765,8 @@ exports[`better eslint`] = {
[0, 0, 0, "Unexpected any. Specify a different type.", "8"] [0, 0, 0, "Unexpected any. Specify a different type.", "8"]
], ],
"public/app/types/store.ts:5381": [ "public/app/types/store.ts:5381": [
[0, 0, 0, "Unexpected any. Specify a different type.", "0"] [0, 0, 0, "Unexpected any. Specify a different type.", "0"],
[0, 0, 0, "Do not use any type assertions.", "1"]
], ],
"public/app/types/templates.ts:5381": [ "public/app/types/templates.ts:5381": [
[0, 0, 0, "Unexpected any. Specify a different type.", "0"] [0, 0, 0, "Unexpected any. Specify a different type.", "0"]

View File

@@ -16,6 +16,7 @@ e2e.scenario({
cy.get('button[title="Delete query"]').each((button) => { cy.get('button[title="Delete query"]').each((button) => {
button.trigger('click'); button.trigger('click');
}); });
cy.get('button[title="Delete query"]').should('not.exist');
e2e.components.QueryTab.queryHistoryButton().should('be.visible').click(); e2e.components.QueryTab.queryHistoryButton().should('be.visible').click();
e2e.components.DataSource.TestData.QueryTab.scenarioSelectContainer() e2e.components.DataSource.TestData.QueryTab.scenarioSelectContainer()
@@ -36,7 +37,7 @@ e2e.scenario({
cy.get('body').click(); cy.get('body').click();
cy.get('body').type('t{leftarrow}'); cy.get('body').type('t{leftarrow}');
cy.location().then((locPostKeypress) => { cy.location().should((locPostKeypress) => {
const params = new URLSearchParams(locPostKeypress.search); const params = new URLSearchParams(locPostKeypress.search);
const leftJSON = JSON.parse(params.get('left')); const leftJSON = JSON.parse(params.get('left'));
// be sure the keypress affected the time window // be sure the keypress affected the time window

View File

@@ -11,7 +11,6 @@ export interface ExploreUrlState<T extends DataQuery = AnyQuery> {
range: RawTimeRange; range: RawTimeRange;
context?: string; context?: string;
panelsState?: ExplorePanelsState; panelsState?: ExplorePanelsState;
isFromCompactUrl?: boolean;
} }
export interface ExplorePanelsState extends Partial<Record<PreferredVisualisationType, {}>> { export interface ExplorePanelsState extends Partial<Record<PreferredVisualisationType, {}>> {

View File

@@ -8,17 +8,13 @@ import { DatasourceSrvMock, MockDataSourceApi } from '../../../test/mocks/dataso
import { import {
buildQueryTransaction, buildQueryTransaction,
clearHistory,
DEFAULT_RANGE, DEFAULT_RANGE,
getRefIds,
getValueWithRefId,
hasNonEmptyQuery, hasNonEmptyQuery,
parseUrlState, parseUrlState,
refreshIntervalToSortOrder, refreshIntervalToSortOrder,
updateHistory, updateHistory,
getExploreUrl, getExploreUrl,
GetExploreUrlArguments, GetExploreUrlArguments,
getTimeRangeFromUrl,
getTimeRange, getTimeRange,
generateEmptyQuery, generateEmptyQuery,
} from './explore'; } from './explore';
@@ -140,7 +136,6 @@ describe('state functions', () => {
const state = { const state = {
...DEFAULT_EXPLORE_STATE, ...DEFAULT_EXPLORE_STATE,
datasource: 'foo', datasource: 'foo',
isFromCompactUrl: false,
queries: [ queries: [
{ {
expr: 'metric{test="a/b"}', expr: 'metric{test="a/b"}',
@@ -165,7 +160,6 @@ describe('state functions', () => {
const state = { const state = {
...DEFAULT_EXPLORE_STATE, ...DEFAULT_EXPLORE_STATE,
datasource: 'foo', datasource: 'foo',
isFromCompactUrl: false,
queries: [ queries: [
{ {
expr: 'metric{test="a/b"}', expr: 'metric{test="a/b"}',
@@ -228,7 +222,7 @@ describe('updateHistory()', () => {
const key = `grafana.explore.history.${datasourceId}`; const key = `grafana.explore.history.${datasourceId}`;
beforeEach(() => { beforeEach(() => {
clearHistory(datasourceId); store.delete(key);
expect(store.exists(key)).toBeFalsy(); expect(store.exists(key)).toBeFalsy();
}); });
@@ -258,87 +252,6 @@ describe('hasNonEmptyQuery', () => {
}); });
}); });
describe('hasRefId', () => {
describe('when called with a null value', () => {
it('then it should return undefined', () => {
const input = null;
const result = getValueWithRefId(input);
expect(result).toBeUndefined();
});
});
describe('when called with a non object value', () => {
it('then it should return undefined', () => {
const input = 123;
const result = getValueWithRefId(input);
expect(result).toBeUndefined();
});
});
describe('when called with an object that has refId', () => {
it('then it should return the object', () => {
const input = { refId: 'A' };
const result = getValueWithRefId(input);
expect(result).toBe(input);
});
});
describe('when called with an array that has refId', () => {
it('then it should return the object', () => {
const input = [123, null, {}, { refId: 'A' }];
const result = getValueWithRefId(input);
expect(result).toBe(input[3]);
});
});
describe('when called with an object that has refId somewhere in the object tree', () => {
it('then it should return the object', () => {
const mockObject = { refId: 'A' };
const input = { data: [123, null, {}, { series: [123, null, {}, mockObject] }] };
const result = getValueWithRefId(input);
expect(result).toBe(mockObject);
});
});
});
describe('getTimeRangeFromUrl', () => {
it('should parse moment date', () => {
// convert date strings to moment object
const range = { from: dateTime('2020-10-22T10:44:33.615Z'), to: dateTime('2020-10-22T10:49:33.615Z') };
const result = getTimeRangeFromUrl(range, 'browser', 0);
expect(result.raw).toEqual(range);
});
it('should parse epoch strings', () => {
const range = {
from: dateTime('2020-10-22T10:00:00Z').valueOf().toString(),
to: dateTime('2020-10-22T11:00:00Z').valueOf().toString(),
};
const result = getTimeRangeFromUrl(range, 'browser', 0);
expect(result.from.valueOf()).toEqual(dateTime('2020-10-22T10:00:00Z').valueOf());
expect(result.to.valueOf()).toEqual(dateTime('2020-10-22T11:00:00Z').valueOf());
expect(result.raw.from.valueOf()).toEqual(dateTime('2020-10-22T10:00:00Z').valueOf());
expect(result.raw.to.valueOf()).toEqual(dateTime('2020-10-22T11:00:00Z').valueOf());
});
it('should parse ISO strings', () => {
const range = {
from: dateTime('2020-10-22T10:00:00Z').toISOString(),
to: dateTime('2020-10-22T11:00:00Z').toISOString(),
};
const result = getTimeRangeFromUrl(range, 'browser', 0);
expect(result.from.valueOf()).toEqual(dateTime('2020-10-22T10:00:00Z').valueOf());
expect(result.to.valueOf()).toEqual(dateTime('2020-10-22T11:00:00Z').valueOf());
expect(result.raw.from.valueOf()).toEqual(dateTime('2020-10-22T10:00:00Z').valueOf());
expect(result.raw.to.valueOf()).toEqual(dateTime('2020-10-22T11:00:00Z').valueOf());
});
});
describe('getTimeRange', () => { describe('getTimeRange', () => {
describe('should flip from and to when from is after to', () => { describe('should flip from and to when from is after to', () => {
const rawRange = { const rawRange = {
@@ -352,62 +265,6 @@ describe('getTimeRange', () => {
}); });
}); });
describe('getRefIds', () => {
describe('when called with a null value', () => {
it('then it should return empty array', () => {
const input = null;
const result = getRefIds(input);
expect(result).toEqual([]);
});
});
describe('when called with a non object value', () => {
it('then it should return empty array', () => {
const input = 123;
const result = getRefIds(input);
expect(result).toEqual([]);
});
});
describe('when called with an object that has refId', () => {
it('then it should return an array with that refId', () => {
const input = { refId: 'A' };
const result = getRefIds(input);
expect(result).toEqual(['A']);
});
});
describe('when called with an array that has refIds', () => {
it('then it should return an array with unique refIds', () => {
const input = [123, null, {}, { refId: 'A' }, { refId: 'A' }, { refId: 'B' }];
const result = getRefIds(input);
expect(result).toEqual(['A', 'B']);
});
});
describe('when called with an object that has refIds somewhere in the object tree', () => {
it('then it should return return an array with unique refIds', () => {
const input = {
data: [
123,
null,
{ refId: 'B', series: [{ refId: 'X' }] },
{ refId: 'B' },
{},
{ series: [123, null, {}, { refId: 'A' }] },
],
};
const result = getRefIds(input);
expect(result).toEqual(['B', 'X', 'A']);
});
});
});
describe('refreshIntervalToSortOrder', () => { describe('refreshIntervalToSortOrder', () => {
describe('when called with live option', () => { describe('when called with live option', () => {
it('then it should return ascending', () => { it('then it should return ascending', () => {

View File

@@ -1,4 +1,4 @@
import { flatten, omit, uniq } from 'lodash'; import { omit } from 'lodash';
import { Unsubscribable } from 'rxjs'; import { Unsubscribable } from 'rxjs';
import { v4 as uuidv4 } from 'uuid'; import { v4 as uuidv4 } from 'uuid';
@@ -8,21 +8,16 @@ import {
DataQueryRequest, DataQueryRequest,
DataSourceApi, DataSourceApi,
DataSourceRef, DataSourceRef,
dateMath,
DateTime,
DefaultTimeZone, DefaultTimeZone,
ExploreUrlState, ExploreUrlState,
HistoryItem, HistoryItem,
IntervalValues, IntervalValues,
isDateTime,
LogsDedupStrategy, LogsDedupStrategy,
LogsSortOrder, LogsSortOrder,
rangeUtil, rangeUtil,
RawTimeRange, RawTimeRange,
TimeFragment,
TimeRange, TimeRange,
TimeZone, TimeZone,
toUtc,
urlUtil, urlUtil,
} from '@grafana/data'; } from '@grafana/data';
import { DataSourceSrv, getDataSourceSrv } from '@grafana/runtime'; import { DataSourceSrv, getDataSourceSrv } from '@grafana/runtime';
@@ -48,8 +43,12 @@ export const DEFAULT_UI_STATE = {
const MAX_HISTORY_ITEMS = 100; const MAX_HISTORY_ITEMS = 100;
export const LAST_USED_DATASOURCE_KEY = 'grafana.explore.datasource'; const LAST_USED_DATASOURCE_KEY = 'grafana.explore.datasource';
export const lastUsedDatasourceKeyForOrgId = (orgId: number) => `${LAST_USED_DATASOURCE_KEY}.${orgId}`; const lastUsedDatasourceKeyForOrgId = (orgId: number) => `${LAST_USED_DATASOURCE_KEY}.${orgId}`;
export const getLastUsedDatasourceUID = (orgId: number) =>
store.getObject<string>(lastUsedDatasourceKeyForOrgId(orgId));
export const setLastUsedDatasourceUID = (orgId: number, datasourceUID: string) =>
store.setObject(lastUsedDatasourceKeyForOrgId(orgId), datasourceUID);
export interface GetExploreUrlArguments { export interface GetExploreUrlArguments {
panel: PanelModel; panel: PanelModel;
@@ -223,8 +222,7 @@ export function parseUrlState(initial: string | undefined): ExploreUrlState {
} }
if (!Array.isArray(parsed)) { if (!Array.isArray(parsed)) {
const urlState = { ...parsed, isFromCompactUrl: false }; return { queries: [], range: DEFAULT_RANGE, ...parsed };
return urlState;
} }
if (parsed.length <= ParseUrlStateIndex.SegmentsStart) { if (parsed.length <= ParseUrlStateIndex.SegmentsStart) {
@@ -241,7 +239,7 @@ export function parseUrlState(initial: string | undefined): ExploreUrlState {
const queries = parsedSegments.filter((segment) => !isSegment(segment, 'ui', 'mode', '__panelsState')); const queries = parsedSegments.filter((segment) => !isSegment(segment, 'ui', 'mode', '__panelsState'));
const panelsState = parsedSegments.find((segment) => isSegment(segment, '__panelsState'))?.__panelsState; const panelsState = parsedSegments.find((segment) => isSegment(segment, '__panelsState'))?.__panelsState;
return { datasource, queries, range, panelsState, isFromCompactUrl: true }; return { datasource, queries, range, panelsState };
} }
export function generateKey(index = 0): string { export function generateKey(index = 0): string {
@@ -285,15 +283,6 @@ export const generateNewKeyAndAddRefIdIfMissing = (target: DataQuery, queries: D
return { ...target, refId, key }; return { ...target, refId, key };
}; };
export const queryDatasourceDetails = (queries: DataQuery[]) => {
const allUIDs = queries.map((query) => query.datasource?.uid);
return {
allHaveDatasource: allUIDs.length === queries.length,
noneHaveDatasource: allUIDs.length === 0,
allDatasourceSame: allUIDs.every((val, i, arr) => val === arr[0]),
};
};
/** /**
* Ensure at least one target exists and that targets have the necessary keys * Ensure at least one target exists and that targets have the necessary keys
* *
@@ -396,11 +385,6 @@ export function updateHistory<T extends DataQuery>(
} }
} }
export function clearHistory(datasourceId: string) {
const historyKey = `grafana.explore.history.${datasourceId}`;
store.delete(historyKey);
}
export const getQueryKeys = (queries: DataQuery[]): string[] => { export const getQueryKeys = (queries: DataQuery[]): string[] => {
const queryKeys = queries.reduce<string[]>((newQueryKeys, query, index) => { const queryKeys = queries.reduce<string[]>((newQueryKeys, query, index) => {
const primaryKey = query.datasource?.uid || query.key; const primaryKey = query.datasource?.uid || query.key;
@@ -420,105 +404,6 @@ export const getTimeRange = (timeZone: TimeZone, rawRange: RawTimeRange, fiscalY
return range; return range;
}; };
const parseRawTime = (value: string | DateTime): TimeFragment | null => {
if (value === null) {
return null;
}
if (isDateTime(value)) {
return value;
}
if (value.indexOf('now') !== -1) {
return value;
}
if (value.length === 8) {
return toUtc(value, 'YYYYMMDD');
}
if (value.length === 15) {
return toUtc(value, 'YYYYMMDDTHHmmss');
}
// Backward compatibility
if (value.length === 19) {
return toUtc(value, 'YYYY-MM-DD HH:mm:ss');
}
// This should handle cases where value is an epoch time as string
if (value.match(/^\d+$/)) {
const epoch = parseInt(value, 10);
return toUtc(epoch);
}
// This should handle ISO strings
const time = toUtc(value);
if (time.isValid()) {
return time;
}
return null;
};
export const getTimeRangeFromUrl = (
range: RawTimeRange,
timeZone: TimeZone,
fiscalYearStartMonth: number
): TimeRange => {
const raw = {
from: parseRawTime(range.from)!,
to: parseRawTime(range.to)!,
};
return {
from: dateMath.parse(raw.from, false, timeZone)!,
to: dateMath.parse(raw.to, true, timeZone)!,
raw,
};
};
export const getValueWithRefId = (value?: any): any => {
if (!value || typeof value !== 'object') {
return undefined;
}
if (value.refId) {
return value;
}
const keys = Object.keys(value);
for (let index = 0; index < keys.length; index++) {
const key = keys[index];
const refId = getValueWithRefId(value[key]);
if (refId) {
return refId;
}
}
return undefined;
};
export const getRefIds = (value: any): string[] => {
if (!value) {
return [];
}
if (typeof value !== 'object') {
return [];
}
const keys = Object.keys(value);
const refIds = [];
for (let index = 0; index < keys.length; index++) {
const key = keys[index];
if (key === 'refId') {
refIds.push(value[key]);
continue;
}
refIds.push(getRefIds(value[key]));
}
return uniq(flatten(refIds));
};
export const refreshIntervalToSortOrder = (refreshInterval?: string) => export const refreshIntervalToSortOrder = (refreshInterval?: string) =>
RefreshPicker.isLive(refreshInterval) ? LogsSortOrder.Ascending : LogsSortOrder.Descending; RefreshPicker.isLive(refreshInterval) ? LogsSortOrder.Ascending : LogsSortOrder.Descending;

View File

@@ -5,11 +5,13 @@ import { TestProvider } from 'test/helpers/TestProvider';
import { DataSourceApi, LoadingState, CoreApp, createTheme, EventBusSrv } from '@grafana/data'; import { DataSourceApi, LoadingState, CoreApp, createTheme, EventBusSrv } from '@grafana/data';
import { selectors } from '@grafana/e2e-selectors'; import { selectors } from '@grafana/e2e-selectors';
import { configureStore } from 'app/store/configureStore';
import { ExploreId } from 'app/types'; import { ExploreId } from 'app/types';
import { Explore, Props } from './Explore'; import { Explore, Props } from './Explore';
import { initialExploreState } from './state/main';
import { scanStopAction } from './state/query'; import { scanStopAction } from './state/query';
import { createEmptyQueryResponse } from './state/utils'; import { createEmptyQueryResponse, makeExplorePaneState } from './state/utils';
const resizeWindow = (x: number, y: number) => { const resizeWindow = (x: number, y: number) => {
global.innerWidth = x; global.innerWidth = x;
@@ -59,7 +61,6 @@ const dummyProps: Props = {
QueryEditorHelp: {}, QueryEditorHelp: {},
}, },
} as DataSourceApi, } as DataSourceApi,
datasourceMissing: false,
exploreId: ExploreId.left, exploreId: ExploreId.left,
loading: false, loading: false,
modifyQueries: jest.fn(), modifyQueries: jest.fn(),
@@ -89,7 +90,6 @@ const dummyProps: Props = {
showFlameGraph: true, showFlameGraph: true,
splitOpen: jest.fn(), splitOpen: jest.fn(),
splitted: false, splitted: false,
isFromCompactUrl: false,
eventBus: new EventBusSrv(), eventBus: new EventBusSrv(),
showRawPrometheus: false, showRawPrometheus: false,
showLogsSample: false, showLogsSample: false,
@@ -119,10 +119,18 @@ jest.mock('react-virtualized-auto-sizer', () => {
}); });
const setup = (overrideProps?: Partial<Props>) => { const setup = (overrideProps?: Partial<Props>) => {
const store = configureStore({
explore: {
...initialExploreState,
panes: {
left: makeExplorePaneState(),
},
},
});
const exploreProps = { ...dummyProps, ...overrideProps }; const exploreProps = { ...dummyProps, ...overrideProps };
return render( return render(
<TestProvider> <TestProvider store={store}>
<Explore {...exploreProps} /> <Explore {...exploreProps} />
</TestProvider> </TestProvider>
); );
@@ -139,8 +147,7 @@ describe('Explore', () => {
}); });
it('should render no data with done loading state', async () => { it('should render no data with done loading state', async () => {
const queryResp = makeEmptyQueryResponse(LoadingState.Done); setup({ queryResponse: makeEmptyQueryResponse(LoadingState.Done) });
setup({ queryResponse: queryResp });
// Wait for the Explore component to render // Wait for the Explore component to render
await screen.findByTestId(selectors.components.DataSourcePicker.container); await screen.findByTestId(selectors.components.DataSourcePicker.container);

View File

@@ -25,18 +25,16 @@ import {
Themeable2, Themeable2,
withTheme2, withTheme2,
PanelContainer, PanelContainer,
Alert,
AdHocFilterItem, AdHocFilterItem,
} from '@grafana/ui'; } from '@grafana/ui';
import { FILTER_FOR_OPERATOR, FILTER_OUT_OPERATOR } from '@grafana/ui/src/components/Table/types'; import { FILTER_FOR_OPERATOR, FILTER_OUT_OPERATOR } from '@grafana/ui/src/components/Table/types';
import appEvents from 'app/core/app_events'; import appEvents from 'app/core/app_events';
import { FadeIn } from 'app/core/components/Animations/FadeIn';
import { supportedFeatures } from 'app/core/history/richHistoryStorageProvider'; import { supportedFeatures } from 'app/core/history/richHistoryStorageProvider';
import { MIXED_DATASOURCE_NAME } from 'app/plugins/datasource/mixed/MixedDataSource'; import { MIXED_DATASOURCE_NAME } from 'app/plugins/datasource/mixed/MixedDataSource';
import { getNodeGraphDataFrames } from 'app/plugins/panel/nodeGraph/utils'; import { getNodeGraphDataFrames } from 'app/plugins/panel/nodeGraph/utils';
import { StoreState } from 'app/types'; import { StoreState } from 'app/types';
import { AbsoluteTimeEvent } from 'app/types/events'; import { AbsoluteTimeEvent } from 'app/types/events';
import { ExploreId, ExploreItemState } from 'app/types/explore'; import { ExploreId } from 'app/types/explore';
import { getTimeZone } from '../profile/state/selectors'; import { getTimeZone } from '../profile/state/selectors';
@@ -285,17 +283,6 @@ export class Explore extends React.PureComponent<Props, ExploreState> {
return <NoData />; return <NoData />;
} }
renderCompactUrlWarning() {
return (
<FadeIn in={true} duration={100}>
<Alert severity="warning" title="Compact URL Deprecation Notice" topSpacing={2}>
The URL that brought you here was a compact URL - this format will soon be deprecated. Please replace the URL
previously saved with the URL available now.
</Alert>
</FadeIn>
);
}
renderGraphPanel(width: number) { renderGraphPanel(width: number) {
const { graphResult, absoluteRange, timeZone, queryResponse, loading, showFlameGraph } = this.props; const { graphResult, absoluteRange, timeZone, queryResponse, loading, showFlameGraph } = this.props;
@@ -426,7 +413,6 @@ export class Explore extends React.PureComponent<Props, ExploreState> {
render() { render() {
const { const {
datasourceInstance, datasourceInstance,
datasourceMissing,
exploreId, exploreId,
graphResult, graphResult,
queryResponse, queryResponse,
@@ -440,7 +426,6 @@ export class Explore extends React.PureComponent<Props, ExploreState> {
showNodeGraph, showNodeGraph,
showFlameGraph, showFlameGraph,
timeZone, timeZone,
isFromCompactUrl,
showLogsSample, showLogsSample,
} = this.props; } = this.props;
const { openDrawer } = this.state; const { openDrawer } = this.state;
@@ -468,9 +453,7 @@ export class Explore extends React.PureComponent<Props, ExploreState> {
scrollRefCallback={(scrollElement) => (this.scrollElement = scrollElement || undefined)} scrollRefCallback={(scrollElement) => (this.scrollElement = scrollElement || undefined)}
> >
<ExploreToolbar exploreId={exploreId} onChangeTime={this.onChangeTime} topOfViewRef={this.topOfViewRef} /> <ExploreToolbar exploreId={exploreId} onChangeTime={this.onChangeTime} topOfViewRef={this.topOfViewRef} />
{isFromCompactUrl ? this.renderCompactUrlWarning() : null} {datasourceInstance ? (
{datasourceMissing ? this.renderEmptyState(styles.exploreContainer) : null}
{datasourceInstance && (
<div className={styles.exploreContainer}> <div className={styles.exploreContainer}>
<PanelContainer className={styles.queryContainer}> <PanelContainer className={styles.queryContainer}>
<QueryRows exploreId={exploreId} /> <QueryRows exploreId={exploreId} />
@@ -537,6 +520,8 @@ export class Explore extends React.PureComponent<Props, ExploreState> {
}} }}
</AutoSizer> </AutoSizer>
</div> </div>
) : (
this.renderEmptyState(styles.exploreContainer)
)} )}
</CustomScrollbar> </CustomScrollbar>
); );
@@ -546,11 +531,11 @@ export class Explore extends React.PureComponent<Props, ExploreState> {
function mapStateToProps(state: StoreState, { exploreId }: ExploreProps) { function mapStateToProps(state: StoreState, { exploreId }: ExploreProps) {
const explore = state.explore; const explore = state.explore;
const { syncedTimes } = explore; const { syncedTimes } = explore;
const item: ExploreItemState = explore.panes[exploreId]!; const item = explore.panes[exploreId]!;
const timeZone = getTimeZone(state.user); const timeZone = getTimeZone(state.user);
const { const {
datasourceInstance, datasourceInstance,
datasourceMissing,
queryKeys, queryKeys,
queries, queries,
isLive, isLive,
@@ -566,7 +551,6 @@ function mapStateToProps(state: StoreState, { exploreId }: ExploreProps) {
showNodeGraph, showNodeGraph,
showFlameGraph, showFlameGraph,
loading, loading,
isFromCompactUrl,
showRawPrometheus, showRawPrometheus,
supplementaryQueries, supplementaryQueries,
} = item; } = item;
@@ -577,7 +561,6 @@ function mapStateToProps(state: StoreState, { exploreId }: ExploreProps) {
return { return {
datasourceInstance, datasourceInstance,
datasourceMissing,
queryKeys, queryKeys,
queries, queries,
isLive, isLive,
@@ -596,7 +579,6 @@ function mapStateToProps(state: StoreState, { exploreId }: ExploreProps) {
showFlameGraph, showFlameGraph,
splitted: isSplit(state), splitted: isSplit(state),
loading, loading,
isFromCompactUrl: isFromCompactUrl || false,
logsSample, logsSample,
showLogsSample, showLogsSample,
}; };

View File

@@ -32,7 +32,7 @@ export const ExploreActions = ({ exploreIdLeft, exploreIdRight }: Props) => {
name: 'Run query (left)', name: 'Run query (left)',
keywords: 'query left', keywords: 'query left',
perform: () => { perform: () => {
dispatch(runQueries(exploreIdLeft)); dispatch(runQueries({ exploreId: exploreIdLeft }));
}, },
section: exploreSection, section: exploreSection,
}); });
@@ -43,7 +43,7 @@ export const ExploreActions = ({ exploreIdLeft, exploreIdRight }: Props) => {
name: 'Run query (right)', name: 'Run query (right)',
keywords: 'query right', keywords: 'query right',
perform: () => { perform: () => {
dispatch(runQueries(exploreIdRight)); dispatch(runQueries({ exploreId: exploreIdRight }));
}, },
section: exploreSection, section: exploreSection,
}); });
@@ -72,7 +72,7 @@ export const ExploreActions = ({ exploreIdLeft, exploreIdRight }: Props) => {
name: 'Run query', name: 'Run query',
keywords: 'query', keywords: 'query',
perform: () => { perform: () => {
dispatch(runQueries(exploreIdLeft)); dispatch(runQueries({ exploreId: exploreIdLeft }));
}, },
section: exploreSection, section: exploreSection,
}); });

View File

@@ -4,12 +4,12 @@ import React, { ComponentProps } from 'react';
import AutoSizer from 'react-virtualized-auto-sizer'; import AutoSizer from 'react-virtualized-auto-sizer';
import { serializeStateToUrlParam } from '@grafana/data'; import { serializeStateToUrlParam } from '@grafana/data';
import { locationService, config } from '@grafana/runtime'; import { config } from '@grafana/runtime';
import { ExploreId } from 'app/types';
import { makeLogsQueryResponse } from './spec/helper/query'; import { makeLogsQueryResponse } from './spec/helper/query';
import { setupExplore, tearDown, waitForExplore } from './spec/helper/setup'; import { setupExplore, tearDown, waitForExplore } from './spec/helper/setup';
import * as mainState from './state/main'; import * as mainState from './state/main';
import * as queryState from './state/query';
jest.mock('app/core/core', () => { jest.mock('app/core/core', () => {
return { return {
@@ -40,13 +40,49 @@ describe('ExplorePage', () => {
describe('Handles open/close splits and related events in UI and URL', () => { describe('Handles open/close splits and related events in UI and URL', () => {
it('opens the split pane when split button is clicked', async () => { it('opens the split pane when split button is clicked', async () => {
setupExplore(); const { location } = setupExplore();
await waitFor(() => {
const editors = screen.getAllByText('loki Editor input:');
expect(editors.length).toBe(1);
// initializing explore replaces the first history entry
expect(location.getHistory().length).toBe(1);
expect(location.getHistory().action).toBe('REPLACE');
});
// Wait for rendering the editor // Wait for rendering the editor
const splitButton = await screen.findByText(/split/i); const splitButton = await screen.findByRole('button', { name: /split/i });
await userEvent.click(splitButton); await userEvent.click(splitButton);
await waitFor(() => { await waitFor(() => {
const editors = screen.getAllByText('loki Editor input:'); const editors = screen.getAllByText('loki Editor input:');
expect(editors.length).toBe(2); expect(editors.length).toBe(2);
// a new entry is pushed to the history
expect(location.getHistory().length).toBe(2);
});
act(() => {
location.getHistory().goBack();
});
await waitFor(() => {
const editors = screen.getAllByText('loki Editor input:');
expect(editors.length).toBe(1);
// going back pops the history
expect(location.getHistory().action).toBe('POP');
expect(location.getHistory().length).toBe(2);
});
act(() => {
location.getHistory().goForward();
});
await waitFor(() => {
const editors = screen.getAllByText('loki Editor input:');
expect(editors.length).toBe(2);
// going forward pops the history
expect(location.getHistory().action).toBe('POP');
expect(location.getHistory().length).toBe(2);
}); });
}); });
@@ -64,7 +100,7 @@ describe('ExplorePage', () => {
}), }),
}; };
const { datasources } = setupExplore({ urlParams }); const { datasources, location } = setupExplore({ urlParams });
jest.mocked(datasources.loki.query).mockReturnValueOnce(makeLogsQueryResponse()); jest.mocked(datasources.loki.query).mockReturnValueOnce(makeLogsQueryResponse());
jest.mocked(datasources.elastic.query).mockReturnValueOnce(makeLogsQueryResponse()); jest.mocked(datasources.elastic.query).mockReturnValueOnce(makeLogsQueryResponse());
@@ -79,14 +115,11 @@ describe('ExplorePage', () => {
expect(logsLines.length).toBe(2); expect(logsLines.length).toBe(2);
// And that the editor gets the expr from the url // And that the editor gets the expr from the url
await screen.findByText(`loki Editor input: { label="value"}`); expect(screen.getByText(`loki Editor input: { label="value"}`)).toBeInTheDocument();
await screen.findByText(`elastic Editor input: error`); expect(screen.getByText(`elastic Editor input: error`)).toBeInTheDocument();
// We did not change the url // We did not change the url
expect(locationService.getSearchObject()).toEqual({ expect(location.getSearchObject()).toEqual(expect.objectContaining(urlParams));
orgId: '1',
...urlParams,
});
// We called the data source query method once // We called the data source query method once
expect(datasources.loki.query).toBeCalledTimes(1); expect(datasources.loki.query).toBeCalledTimes(1);
@@ -100,39 +133,50 @@ describe('ExplorePage', () => {
}); });
}); });
// TODO: the following tests are using the compact format, we should use the current format instead
// and have a dedicated test ensuring the compact format is parsed correctly
it('can close a panel from a split', async () => { it('can close a panel from a split', async () => {
const urlParams = { const urlParams = {
left: JSON.stringify(['now-1h', 'now', 'loki', { refId: 'A' }]), left: JSON.stringify(['now-1h', 'now', 'loki', { refId: 'A' }]),
right: JSON.stringify(['now-1h', 'now', 'elastic', { refId: 'A' }]), right: JSON.stringify(['now-1h', 'now', 'elastic', { refId: 'A' }]),
}; };
setupExplore({ urlParams }); const { location } = setupExplore({ urlParams });
const closeButtons = await screen.findAllByLabelText(/Close split pane/i); let closeButtons = await screen.findAllByLabelText(/Close split pane/i);
await userEvent.click(closeButtons[1]); await userEvent.click(closeButtons[1]);
expect(location.getHistory().length).toBe(1);
await waitFor(() => { await waitFor(() => {
const postCloseButtons = screen.queryAllByLabelText(/Close split pane/i); closeButtons = screen.queryAllByLabelText(/Close split pane/i);
expect(postCloseButtons.length).toBe(0); expect(closeButtons.length).toBe(0);
// Closing a pane using the split close button causes a new entry to be pushed in the history
expect(location.getHistory().length).toBe(2);
}); });
}); });
it('handles url change to split view', async () => { it('Reacts to URL changes and opens a pane if an entry is pushed to history', async () => {
const urlParams = { const urlParams = {
left: JSON.stringify(['now-1h', 'now', 'loki', { expr: '{ label="value"}' }]), left: JSON.stringify(['now-1h', 'now', 'loki', { expr: '{ label="value"}' }]),
}; };
const { datasources } = setupExplore({ urlParams }); const { datasources, location } = setupExplore({ urlParams });
jest.mocked(datasources.loki.query).mockReturnValue(makeLogsQueryResponse()); jest.mocked(datasources.loki.query).mockReturnValue(makeLogsQueryResponse());
jest.mocked(datasources.elastic.query).mockReturnValue(makeLogsQueryResponse()); jest.mocked(datasources.elastic.query).mockReturnValue(makeLogsQueryResponse());
await waitFor(() => {
expect(screen.getByText(`loki Editor input: { label="value"}`)).toBeInTheDocument();
});
act(() => { act(() => {
locationService.partial({ location.partial({
left: JSON.stringify(['now-1h', 'now', 'loki', { expr: '{ label="value"}' }]), left: JSON.stringify(['now-1h', 'now', 'loki', { expr: '{ label="value"}' }]),
right: JSON.stringify(['now-1h', 'now', 'elastic', { expr: 'error' }]), right: JSON.stringify(['now-1h', 'now', 'elastic', { expr: 'error' }]),
}); });
}); });
// Editor renders the new query await waitFor(() => {
await screen.findByText(`loki Editor input: { label="value"}`); expect(screen.getByText(`loki Editor input: { label="value"}`)).toBeInTheDocument();
await screen.findByText(`elastic Editor input: error`); expect(screen.getByText(`elastic Editor input: error`)).toBeInTheDocument();
});
}); });
it('handles opening split with split open func', async () => { it('handles opening split with split open func', async () => {
@@ -143,40 +187,42 @@ describe('ExplorePage', () => {
jest.mocked(datasources.loki.query).mockReturnValue(makeLogsQueryResponse()); jest.mocked(datasources.loki.query).mockReturnValue(makeLogsQueryResponse());
jest.mocked(datasources.elastic.query).mockReturnValue(makeLogsQueryResponse()); jest.mocked(datasources.elastic.query).mockReturnValue(makeLogsQueryResponse());
// This is mainly to wait for render so that the left pane state is initialized as that is needed for splitOpen // Wait for the left pane to render
// to work await waitFor(async () => {
await screen.findByText(`loki Editor input: { label="value"}`); expect(await screen.findByText(`loki Editor input: { label="value"}`)).toBeInTheDocument();
});
act(() => { act(() => {
store.dispatch(mainState.splitOpen({ datasourceUid: 'elastic', query: { expr: 'error', refId: 'A' } })); store.dispatch(mainState.splitOpen({ datasourceUid: 'elastic', query: { expr: 'error', refId: 'A' } }));
}); });
// Editor renders the new query // Editor renders the new query
await screen.findByText(`elastic Editor input: error`); expect(await screen.findByText(`elastic Editor input: error`)).toBeInTheDocument();
await screen.findByText(`loki Editor input: { label="value"}`); expect(await screen.findByText(`loki Editor input: { label="value"}`)).toBeInTheDocument();
}); });
it('handles split size events and sets relevant variables', async () => { it('handles split size events and sets relevant variables', async () => {
setupExplore(); setupExplore();
const splitButton = await screen.findByText(/split/i); const splitButton = await screen.findByText(/split/i);
await userEvent.click(splitButton); await userEvent.click(splitButton);
await waitForExplore(undefined, true); await waitForExplore(ExploreId.left);
let widenButton = await screen.findAllByLabelText('Widen pane');
let narrowButton = await screen.queryAllByLabelText('Narrow pane'); expect(await screen.findAllByLabelText('Widen pane')).toHaveLength(2);
expect(screen.queryByLabelText('Narrow pane')).not.toBeInTheDocument();
const panes = screen.getAllByRole('main'); const panes = screen.getAllByRole('main');
expect(widenButton.length).toBe(2);
expect(narrowButton.length).toBe(0);
expect(Number.parseInt(getComputedStyle(panes[0]).width, 10)).toBe(1000); expect(Number.parseInt(getComputedStyle(panes[0]).width, 10)).toBe(1000);
expect(Number.parseInt(getComputedStyle(panes[1]).width, 10)).toBe(1000); expect(Number.parseInt(getComputedStyle(panes[1]).width, 10)).toBe(1000);
const resizer = screen.getByRole('presentation'); const resizer = screen.getByRole('presentation');
fireEvent.mouseDown(resizer, { buttons: 1 }); fireEvent.mouseDown(resizer, { buttons: 1 });
fireEvent.mouseMove(resizer, { clientX: -700, buttons: 1 }); fireEvent.mouseMove(resizer, { clientX: -700, buttons: 1 });
fireEvent.mouseUp(resizer); fireEvent.mouseUp(resizer);
widenButton = await screen.findAllByLabelText('Widen pane');
narrowButton = await screen.queryAllByLabelText('Narrow pane'); expect(await screen.findAllByLabelText('Widen pane')).toHaveLength(1);
expect(widenButton.length).toBe(1); expect(await screen.findAllByLabelText('Narrow pane')).toHaveLength(1);
expect(narrowButton.length).toBe(1);
// the autosizer is mocked so there is no actual resize here
}); });
}); });
@@ -214,205 +260,241 @@ describe('ExplorePage', () => {
}); });
describe('Handles different URL datasource redirects', () => { describe('Handles different URL datasource redirects', () => {
it('No params, no store value uses default data source', async () => { describe('exploreMixedDatasource on', () => {
setupExplore(); beforeAll(() => {
await waitForExplore();
const urlParams = decodeURIComponent(locationService.getSearch().toString());
expect(urlParams).toBe(
'orgId=1&left={"datasource":"loki-uid","queries":[{"refId":"A","datasource":{"type":"logs","uid":"loki-uid"}}],"range":{"from":"now-1h","to":"now"}}'
);
});
it('No datasource in root or query and no store value uses default data source', async () => {
setupExplore({ urlParams: 'orgId=1&left={"queries":[{"refId":"A"}],"range":{"from":"now-1h","to":"now"}}' });
await waitForExplore();
const urlParams = decodeURIComponent(locationService.getSearch().toString());
expect(urlParams).toBe(
'orgId=1&left={"datasource":"loki-uid","queries":[{"refId":"A"}],"range":{"from":"now-1h","to":"now"}}'
);
});
it('No datasource in root or query with store value uses store value data source', async () => {
setupExplore({
urlParams: 'orgId=1&left={"queries":[{"refId":"A"}],"range":{"from":"now-1h","to":"now"}}',
prevUsedDatasource: { orgId: 1, datasource: 'elastic' },
});
await waitForExplore();
const urlParams = decodeURIComponent(locationService.getSearch().toString());
expect(urlParams).toBe(
'orgId=1&left={"datasource":"elastic-uid","queries":[{"refId":"A"}],"range":{"from":"now-1h","to":"now"}}'
);
});
it('UID datasource in root uses root data source', async () => {
setupExplore({
urlParams:
'orgId=1&left={"datasource":"loki-uid","queries":[{"refId":"A"}],"range":{"from":"now-1h","to":"now"}}',
prevUsedDatasource: { orgId: 1, datasource: 'elastic' },
});
await waitForExplore();
const urlParams = decodeURIComponent(locationService.getSearch().toString());
expect(urlParams).toBe(
'orgId=1&left={"datasource":"loki-uid","queries":[{"refId":"A"}],"range":{"from":"now-1h","to":"now"}}'
);
});
it('Name datasource in root uses root data source, converts to UID', async () => {
setupExplore({
urlParams: 'orgId=1&left={"datasource":"loki","queries":[{"refId":"A"}],"range":{"from":"now-1h","to":"now"}}',
prevUsedDatasource: { orgId: 1, datasource: 'elastic' },
});
await waitForExplore();
const urlParams = decodeURIComponent(locationService.getSearch().toString());
expect(urlParams).toBe(
'orgId=1&left={"datasource":"loki-uid","queries":[{"refId":"A"}],"range":{"from":"now-1h","to":"now"}}'
);
});
it('Datasource ref in query, none in root uses query datasource', async () => {
setupExplore({
urlParams:
'orgId=1&left={"queries":[{"refId":"A","datasource":{"type":"logs","uid":"loki-uid"}}],"range":{"from":"now-1h","to":"now"}}',
prevUsedDatasource: { orgId: 1, datasource: 'elastic' },
});
await waitForExplore();
const urlParams = decodeURIComponent(locationService.getSearch().toString());
expect(urlParams).toBe(
'orgId=1&left={"datasource":"loki-uid","queries":[{"refId":"A","datasource":{"type":"logs","uid":"loki-uid"}}],"range":{"from":"now-1h","to":"now"}}'
);
});
it('Datasource ref in query with matching UID in root uses matching datasource', async () => {
setupExplore({
urlParams:
'orgId=1&left={"datasource":"loki-uid","queries":[{"refId":"A","datasource":{"type":"logs","uid":"loki-uid"}}],"range":{"from":"now-1h","to":"now"}}',
prevUsedDatasource: { orgId: 1, datasource: 'elastic' },
});
await waitForExplore();
const urlParams = decodeURIComponent(locationService.getSearch().toString());
expect(urlParams).toBe(
'orgId=1&left={"datasource":"loki-uid","queries":[{"refId":"A","datasource":{"type":"logs","uid":"loki-uid"}}],"range":{"from":"now-1h","to":"now"}}'
);
});
it('Datasource ref in query with matching name in root uses matching datasource, converts root to UID', async () => {
setupExplore({
urlParams:
'orgId=1&left={"datasource":"loki","queries":[{"refId":"A","datasource":{"type":"logs","uid":"loki-uid"}}],"range":{"from":"now-1h","to":"now"}}',
prevUsedDatasource: { orgId: 1, datasource: 'elastic' },
});
await waitForExplore();
const urlParams = decodeURIComponent(locationService.getSearch().toString());
expect(urlParams).toBe(
'orgId=1&left={"datasource":"loki-uid","queries":[{"refId":"A","datasource":{"type":"logs","uid":"loki-uid"}}],"range":{"from":"now-1h","to":"now"}}'
);
});
it('Datasource ref in query with mismatching UID in root uses query datasource', async () => {
setupExplore({
urlParams:
'orgId=1&left={"datasource":"elastic-uid","queries":[{"refId":"A","datasource":{"type":"logs","uid":"loki-uid"}}],"range":{"from":"now-1h","to":"now"}}',
prevUsedDatasource: { orgId: 1, datasource: 'elastic' },
});
await waitForExplore();
const urlParams = decodeURIComponent(locationService.getSearch().toString());
expect(urlParams).toBe(
'orgId=1&left={"datasource":"loki-uid","queries":[{"refId":"A","datasource":{"type":"logs","uid":"loki-uid"}}],"range":{"from":"now-1h","to":"now"}}'
);
});
it('Different datasources in query with mixed feature on changes root to Mixed', async () => {
config.featureToggles.exploreMixedDatasource = true; config.featureToggles.exploreMixedDatasource = true;
setupExplore({
urlParams:
'orgId=1&left={"datasource":"elastic-uid","queries":[{"refId":"A","datasource":{"type":"logs","uid":"loki-uid"}},{"refId":"B","datasource":{"type":"logs","uid":"elastic-uid"}}],"range":{"from":"now-1h","to":"now"}}',
prevUsedDatasource: { orgId: 1, datasource: 'elastic' },
}); });
const reducerMock = jest.spyOn(queryState, 'queryReducer');
await waitForExplore(undefined, true); describe('When root datasource is not specified in the URL', () => {
const urlParams = decodeURIComponent(locationService.getSearch().toString()); it('Redirects to default datasource', async () => {
expect(reducerMock).not.toHaveBeenCalledWith( const { location } = setupExplore({ mixedEnabled: true });
expect.anything(), await waitForExplore();
expect.objectContaining({ type: 'explore/queriesImported' })
); await waitFor(() => {
// this mixed UID is weird just because of our fake datasource generator const urlParams = decodeURIComponent(location.getSearch().toString());
expect(urlParams).toBe( expect(urlParams).toBe(
'orgId=1&left={"datasource":"--+Mixed+---uid","queries":[{"refId":"A","datasource":{"type":"logs","uid":"loki-uid"}},{"refId":"B","datasource":{"type":"logs","uid":"elastic-uid"}}],"range":{"from":"now-1h","to":"now"}}' 'left={"datasource":"loki-uid","queries":[{"refId":"A","datasource":{"type":"logs","uid":"loki-uid"}}],"range":{"from":"now-1h","to":"now"}}&orgId=1'
); );
});
config.featureToggles.exploreMixedDatasource = false; expect(location.getHistory()).toHaveLength(1);
}); });
it('Different datasources in query with mixed feature off uses first query DS, converts rest', async () => { it('Redirects to last used datasource when available', async () => {
config.featureToggles.exploreMixedDatasource = false; const { location } = setupExplore({
setupExplore({ prevUsedDatasource: { orgId: 1, datasource: 'elastic-uid' },
urlParams: mixedEnabled: true,
'orgId=1&left={"datasource":"elastic-uid","queries":[{"refId":"A","datasource":{"type":"logs","uid":"loki-uid"}},{"refId":"B","datasource":{"type":"logs","uid":"elastic-uid"}}],"range":{"from":"now-1h","to":"now"}}', });
prevUsedDatasource: { orgId: 1, datasource: 'elastic' }, await waitForExplore();
await waitFor(() => {
const urlParams = decodeURIComponent(location.getSearch().toString());
expect(urlParams).toBe(
'left={"datasource":"elastic-uid","queries":[{"refId":"A","datasource":{"type":"logs","uid":"elastic-uid"}}],"range":{"from":"now-1h","to":"now"}}&orgId=1'
);
});
expect(location.getHistory()).toHaveLength(1);
}); });
const reducerMock = jest.spyOn(queryState, 'queryReducer'); it("Redirects to first query's datasource", async () => {
await waitForExplore(undefined, true); const { location } = setupExplore({
const urlParams = decodeURIComponent(locationService.getSearch().toString()); urlParams: {
// because there are no import/export queries in our mock datasources, only the first one remains left: '{"queries":[{"refId":"A","datasource":{"type":"logs","uid":"loki-uid"}}],"range":{"from":"now-1h","to":"now"}}',
expect(reducerMock).toHaveBeenCalledWith(
expect.anything(),
expect.objectContaining({
type: 'explore/queriesImported',
payload: expect.objectContaining({
exploreId: 'left',
queries: [
expect.objectContaining({
datasource: {
type: 'logs',
uid: 'loki-uid',
}, },
}),
],
}),
})
);
expect(urlParams).toBe(
'orgId=1&left={"datasource":"loki-uid","queries":[{"refId":"A","datasource":{"type":"logs","uid":"loki-uid"}}],"range":{"from":"now-1h","to":"now"}}'
);
});
it('Datasource in root not found and no queries changes to default', async () => {
setupExplore({
urlParams: 'orgId=1&left={"datasource":"asdasdasd","range":{"from":"now-1h","to":"now"}}',
prevUsedDatasource: { orgId: 1, datasource: 'elastic' }, prevUsedDatasource: { orgId: 1, datasource: 'elastic' },
mixedEnabled: true,
}); });
await waitForExplore(); await waitForExplore();
const urlParams = decodeURIComponent(locationService.getSearch().toString());
await waitFor(() => {
const urlParams = decodeURIComponent(location.getSearch().toString());
expect(urlParams).toBe( expect(urlParams).toBe(
'orgId=1&left={"datasource":"loki-uid","queries":[{"refId":"A","datasource":{"type":"logs","uid":"loki-uid"}}],"range":{"from":"now-1h","to":"now"}}' 'left={"datasource":"loki-uid","queries":[{"refId":"A","datasource":{"type":"logs","uid":"loki-uid"}}],"range":{"from":"now-1h","to":"now"}}&orgId=1'
); );
}); });
expect(location.getHistory()).toHaveLength(1);
});
});
it('Datasource root is mixed and there are two queries, one with datasource not found, only one query remains with root datasource as that datasource', async () => { describe('When root datasource is specified in the URL', () => {
const consoleErrorSpy = jest.spyOn(console, 'error').mockImplementation(); it('Uses the datasource in the URL', async () => {
setupExplore({ const { location } = setupExplore({
urlParams: urlParams: {
'orgId=1&left={"datasource":"-- Mixed --","queries":[{"refId":"A","datasource":{"type":"asdf","uid":"asdf"}},{"refId":"B","datasource":{"type":"logs","uid":"elastic-uid"}}],"range":{"from":"now-1h","to":"now"}}', left: '{"datasource":"elastic-uid","queries":[{"refId":"A"}],"range":{"from":"now-1h","to":"now"}}',
},
prevUsedDatasource: { orgId: 1, datasource: 'elastic' }, prevUsedDatasource: { orgId: 1, datasource: 'elastic' },
mixedEnabled: true,
}); });
await waitForExplore(); await waitForExplore();
const urlParams = decodeURIComponent(locationService.getSearch().toString());
expect(urlParams).toBe(
'orgId=1&left={"datasource":"elastic-uid","queries":[{"refId":"B","datasource":{"type":"logs","uid":"elastic-uid"}}],"range":{"from":"now-1h","to":"now"}}'
);
expect(consoleErrorSpy).toBeCalledTimes(1);
consoleErrorSpy.mockRestore(); await waitFor(() => {
const urlParams = decodeURIComponent(location.getSearch().toString());
expect(urlParams).toBe(
'left={"datasource":"elastic-uid","queries":[{"refId":"A"}],"range":{"from":"now-1h","to":"now"}}&orgId=1'
);
});
expect(location.getHistory()).toHaveLength(1);
});
it('Filters out queries not using the root datasource', async () => {
const { location } = setupExplore({
urlParams: {
left: '{"datasource":"elastic-uid","queries":[{"refId":"A","datasource":{"type":"logs","uid":"loki-uid"}},{"refId":"B","datasource":{"type":"logs","uid":"elastic-uid"}}],"range":{"from":"now-1h","to":"now"}}',
},
prevUsedDatasource: { orgId: 1, datasource: 'elastic' },
mixedEnabled: true,
});
await waitForExplore();
await waitFor(() => {
const urlParams = decodeURIComponent(location.getSearch().toString());
expect(urlParams).toBe(
'left={"datasource":"elastic-uid","queries":[{"refId":"B","datasource":{"type":"logs","uid":"elastic-uid"}}],"range":{"from":"now-1h","to":"now"}}&orgId=1'
);
});
});
it('Fallbacks to last used datasource if root datasource does not exist', async () => {
const { location } = setupExplore({
urlParams: { left: '{"datasource":"NON-EXISTENT","range":{"from":"now-1h","to":"now"}}' },
prevUsedDatasource: { orgId: 1, datasource: 'elastic' },
mixedEnabled: true,
});
await waitForExplore();
await waitFor(() => {
const urlParams = decodeURIComponent(location.getSearch().toString());
expect(urlParams).toBe(
'left={"datasource":"elastic-uid","queries":[{"refId":"A","datasource":{"type":"logs","uid":"elastic-uid"}}],"range":{"from":"now-1h","to":"now"}}&orgId=1'
);
});
});
it('Fallbacks to default datasource if root datasource does not exist and last used datasource does not exist', async () => {
const { location } = setupExplore({
urlParams: { left: '{"datasource":"NON-EXISTENT","range":{"from":"now-1h","to":"now"}}' },
prevUsedDatasource: { orgId: 1, datasource: 'I DO NOT EXIST' },
mixedEnabled: true,
});
await waitForExplore();
await waitFor(() => {
const urlParams = decodeURIComponent(location.getSearch().toString());
expect(urlParams).toBe(
'left={"datasource":"loki-uid","queries":[{"refId":"A","datasource":{"type":"logs","uid":"loki-uid"}}],"range":{"from":"now-1h","to":"now"}}&orgId=1'
);
});
});
it('Fallbacks to default datasource if root datasource does not exist there is no last used datasource', async () => {
const { location } = setupExplore({
urlParams: { left: '{"datasource":"NON-EXISTENT","range":{"from":"now-1h","to":"now"}}' },
mixedEnabled: true,
});
await waitForExplore();
await waitFor(() => {
const urlParams = decodeURIComponent(location.getSearch().toString());
expect(urlParams).toBe(
'left={"datasource":"loki-uid","queries":[{"refId":"A","datasource":{"type":"logs","uid":"loki-uid"}}],"range":{"from":"now-1h","to":"now"}}&orgId=1'
);
});
});
});
it('Queries using nonexisting datasources gets removed', async () => {
const { location } = setupExplore({
urlParams: {
left: '{"datasource":"-- Mixed --","queries":[{"refId":"A","datasource":{"type":"NON-EXISTENT","uid":"NON-EXISTENT"}},{"refId":"B","datasource":{"type":"logs","uid":"elastic-uid"}}],"range":{"from":"now-1h","to":"now"}}',
},
prevUsedDatasource: { orgId: 1, datasource: 'elastic' },
mixedEnabled: true,
});
await waitForExplore();
await waitFor(() => {
const urlParams = decodeURIComponent(location.getSearch().toString());
expect(urlParams).toBe(
'left={"datasource":"--+Mixed+--","queries":[{"refId":"B","datasource":{"type":"logs","uid":"elastic-uid"}}],"range":{"from":"now-1h","to":"now"}}&orgId=1'
);
});
});
it('Only keeps queries using root datasource', async () => {
const { location } = setupExplore({
urlParams: {
left: '{"datasource":"elastic-uid","queries":[{"refId":"A","datasource":{"type":"logs","uid":"loki-uid"}},{"refId":"B","datasource":{"type":"logs","uid":"elastic-uid"}}],"range":{"from":"now-1h","to":"now"}}',
},
prevUsedDatasource: { orgId: 1, datasource: 'elastic' },
mixedEnabled: true,
});
await waitForExplore();
await waitFor(() => {
const urlParams = decodeURIComponent(location.getSearch().toString());
// because there are no import/export queries in our mock datasources, only the first one remains
expect(urlParams).toBe(
'left={"datasource":"elastic-uid","queries":[{"refId":"B","datasource":{"type":"logs","uid":"elastic-uid"}}],"range":{"from":"now-1h","to":"now"}}&orgId=1'
);
});
});
});
});
describe('exploreMixedDatasource off', () => {
beforeAll(() => {
config.featureToggles.exploreMixedDatasource = false;
});
it('Redirects to the first query datasource if the root is mixed', async () => {
const { location } = setupExplore({
urlParams: {
left: '{"datasource":"-- Mixed --","queries":[{"refId":"A","datasource":{"type":"logs","uid":"elastic-uid"}},{"refId":"B","datasource":{"type":"logs","uid":"loki-uid"}}],"range":{"from":"now-1h","to":"now"}}',
},
mixedEnabled: false,
});
await waitForExplore();
await waitFor(() => {
const urlParams = decodeURIComponent(location.getSearch().toString());
expect(urlParams).toBe(
'left={"datasource":"elastic-uid","queries":[{"refId":"A","datasource":{"type":"logs","uid":"elastic-uid"}}],"range":{"from":"now-1h","to":"now"}}&orgId=1'
);
});
});
it('Redirects to the default datasource if the root is mixed and there are no queries', async () => {
const { location } = setupExplore({
urlParams: {
left: '{"datasource":"-- Mixed --","range":{"from":"now-1h","to":"now"}}',
},
mixedEnabled: false,
});
await waitForExplore();
await waitFor(() => {
const urlParams = decodeURIComponent(location.getSearch().toString());
expect(urlParams).toBe(
'left={"datasource":"loki-uid","queries":[{"refId":"A","datasource":{"type":"logs","uid":"loki-uid"}}],"range":{"from":"now-1h","to":"now"}}&orgId=1'
);
});
}); });
}); });
it('removes `from` and `to` parameters from url when first mounted', async () => { it('removes `from` and `to` parameters from url when first mounted', async () => {
setupExplore({ searchParams: 'from=1&to=2&orgId=1' }); const { location } = setupExplore({ urlParams: { from: '1', to: '2' } });
await waitForExplore(); await waitForExplore();
expect(locationService.getSearchObject()).toEqual(expect.not.objectContaining({ from: '1', to: '2' })); await waitFor(() => {
expect(locationService.getSearchObject()).toEqual(expect.objectContaining({ orgId: '1' })); expect(location.getSearchObject()).toEqual(expect.not.objectContaining({ from: '1', to: '2' }));
expect(location.getSearchObject()).toEqual(expect.objectContaining({ orgId: '1' }));
});
}); });
}); });

View File

@@ -1,25 +1,25 @@
import { css } from '@emotion/css'; import { css } from '@emotion/css';
import { inRange } from 'lodash'; import { inRange } from 'lodash';
import React, { useEffect, useRef, useState } from 'react'; import React, { useEffect, useState } from 'react';
import { useWindowSize } from 'react-use'; import { useWindowSize } from 'react-use';
import { isTruthy } from '@grafana/data'; import { ErrorBoundaryAlert } from '@grafana/ui';
import { locationService } from '@grafana/runtime';
import { ErrorBoundaryAlert, usePanelContext } from '@grafana/ui';
import { SplitPaneWrapper } from 'app/core/components/SplitPaneWrapper/SplitPaneWrapper'; import { SplitPaneWrapper } from 'app/core/components/SplitPaneWrapper/SplitPaneWrapper';
import { useGrafana } from 'app/core/context/GrafanaContext'; import { useGrafana } from 'app/core/context/GrafanaContext';
import { useAppNotification } from 'app/core/copy/appNotification';
import { useNavModel } from 'app/core/hooks/useNavModel'; import { useNavModel } from 'app/core/hooks/useNavModel';
import { GrafanaRouteComponentProps } from 'app/core/navigation/types'; import { GrafanaRouteComponentProps } from 'app/core/navigation/types';
import { useDispatch, useSelector } from 'app/types'; import { useDispatch, useSelector } from 'app/types';
import { ExploreId, ExploreQueryParams } from 'app/types/explore'; import { ExploreId, ExploreQueryParams } from 'app/types/explore';
import { Branding } from '../../core/components/Branding/Branding';
import { useCorrelations } from '../correlations/useCorrelations';
import { ExploreActions } from './ExploreActions'; import { ExploreActions } from './ExploreActions';
import { ExplorePaneContainer } from './ExplorePaneContainer'; import { ExplorePaneContainer } from './ExplorePaneContainer';
import { lastSavedUrl, saveCorrelationsAction, resetExploreAction, splitSizeUpdateAction } from './state/main'; import { useExploreCorrelations } from './hooks/useExploreCorrelations';
import { useExplorePageTitle } from './hooks/useExplorePageTitle';
import { useStateSync } from './hooks/useStateSync';
import { useStopQueries } from './hooks/useStopQueries';
import { useTimeSrvFix } from './hooks/useTimeSrvFix';
import { splitSizeUpdateAction } from './state/main';
import { selectOrderedExplorePanes } from './state/selectors';
const styles = { const styles = {
pageScrollbarWrapper: css` pageScrollbarWrapper: css`
@@ -32,20 +32,21 @@ const styles = {
}; };
export default function ExplorePage(props: GrafanaRouteComponentProps<{}, ExploreQueryParams>) { export default function ExplorePage(props: GrafanaRouteComponentProps<{}, ExploreQueryParams>) {
useStopQueries();
useTimeSrvFix();
useStateSync(props.queryParams);
useExplorePageTitle(); useExplorePageTitle();
useExploreCorrelations();
const dispatch = useDispatch(); const dispatch = useDispatch();
const queryParams = props.queryParams; const { keybindings, chrome } = useGrafana();
const { keybindings, chrome, config } = useGrafana();
const navModel = useNavModel('explore'); const navModel = useNavModel('explore');
const { get } = useCorrelations();
const { warning } = useAppNotification();
const panelCtx = usePanelContext();
const eventBus = useRef(panelCtx.eventBus.newScopedBus('explore', { onlyLocal: false }));
const [rightPaneWidthRatio, setRightPaneWidthRatio] = useState(0.5); const [rightPaneWidthRatio, setRightPaneWidthRatio] = useState(0.5);
const { width: windowWidth } = useWindowSize(); const { width: windowWidth } = useWindowSize();
const minWidth = 200; const minWidth = 200;
const exploreState = useSelector((state) => state.explore); const exploreState = useSelector((state) => state.explore);
const panes = useSelector(selectOrderedExplorePanes);
useEffect(() => { useEffect(() => {
//This is needed for breadcrumbs and topnav. //This is needed for breadcrumbs and topnav.
//We should probably abstract this out at some point //We should probably abstract this out at some point
@@ -56,52 +57,6 @@ export default function ExplorePage(props: GrafanaRouteComponentProps<{}, Explor
keybindings.setupTimeRangeBindings(false); keybindings.setupTimeRangeBindings(false);
}, [keybindings]); }, [keybindings]);
useEffect(() => {
if (!config.featureToggles.correlations) {
dispatch(saveCorrelationsAction([]));
} else {
get.execute();
}
// eslint-disable-next-line react-hooks/exhaustive-deps
}, []);
useEffect(() => {
if (get.value) {
dispatch(saveCorrelationsAction(get.value));
} else if (get.error) {
dispatch(saveCorrelationsAction([]));
warning(
'Could not load correlations.',
'Correlations data could not be loaded, DataLinks may have partial data.'
);
}
}, [get.value, get.error, dispatch, warning]);
useEffect(() => {
lastSavedUrl.left = undefined;
lastSavedUrl.right = undefined;
// timeSrv (which is used internally) on init reads `from` and `to` param from the URL and updates itself
// using those value regardless of what is passed to the init method.
// The updated value is then used by Explore to get the range for each pane.
// This means that if `from` and `to` parameters are present in the URL,
// it would be impossible to change the time range in Explore.
// We are only doing this on mount for 2 reasons:
// 1: Doing it on update means we'll enter a render loop.
// 2: when parsing time in Explore (before feeding it to timeSrv) we make sure `from` is before `to` inside
// each pane state in order to not trigger un URL update from timeSrv.
const searchParams = locationService.getSearchObject();
if (searchParams.from || searchParams.to) {
locationService.partial({ from: undefined, to: undefined }, true);
}
return () => {
// Cleaning up Explore state so that when navigating back to Explore it starts from a blank state
dispatch(resetExploreAction());
};
// eslint-disable-next-line react-hooks/exhaustive-deps -- dispatch is stable, doesn't need to be in the deps array
}, []);
const updateSplitSize = (size: number) => { const updateSplitSize = (size: number) => {
const evenSplitWidth = windowWidth / 2; const evenSplitWidth = windowWidth / 2;
const areBothSimilar = inRange(size, evenSplitWidth - 100, evenSplitWidth + 100); const areBothSimilar = inRange(size, evenSplitWidth - 100, evenSplitWidth + 100);
@@ -118,7 +73,7 @@ export default function ExplorePage(props: GrafanaRouteComponentProps<{}, Explor
setRightPaneWidthRatio(size / windowWidth); setRightPaneWidthRatio(size / windowWidth);
}; };
const hasSplit = Boolean(queryParams.left) && Boolean(queryParams.right); const hasSplit = Object.entries(panes).length > 1;
let widthCalc = 0; let widthCalc = 0;
if (hasSplit) { if (hasSplit) {
if (!exploreState.evenSplitPanes && exploreState.maxedExploreId) { if (!exploreState.evenSplitPanes && exploreState.maxedExploreId) {
@@ -148,30 +103,14 @@ export default function ExplorePage(props: GrafanaRouteComponentProps<{}, Explor
} }
}} }}
> >
<ErrorBoundaryAlert style="page"> {Object.keys(panes).map((exploreId) => {
<ExplorePaneContainer exploreId={ExploreId.left} urlQuery={queryParams.left} eventBus={eventBus.current} /> return (
<ErrorBoundaryAlert key={exploreId} style="page">
<ExplorePaneContainer exploreId={exploreId as ExploreId} />
</ErrorBoundaryAlert> </ErrorBoundaryAlert>
{hasSplit && ( );
<ErrorBoundaryAlert style="page"> })}
<ExplorePaneContainer
exploreId={ExploreId.right}
urlQuery={queryParams.right}
eventBus={eventBus.current}
/>
</ErrorBoundaryAlert>
)}
</SplitPaneWrapper> </SplitPaneWrapper>
</div> </div>
); );
} }
const useExplorePageTitle = () => {
const navModel = useNavModel('explore');
const datasources = useSelector((state) =>
[state.explore.panes.left!.datasourceInstance?.name, state.explore.panes.right?.datasourceInstance?.name].filter(
isTruthy
)
);
document.title = `${navModel.main.text} - ${datasources.join(' | ')} - ${Branding.AppTitle}`;
};

View File

@@ -1,35 +1,14 @@
import { css } from '@emotion/css'; import { css } from '@emotion/css';
import memoizeOne from 'memoize-one'; import React, { useEffect, useRef } from 'react';
import React from 'react'; import { connect } from 'react-redux';
import { connect, ConnectedProps } from 'react-redux';
import { EventBusExtended, EventBusSrv, GrafanaTheme2, EventBus } from '@grafana/data'; import { EventBusSrv, GrafanaTheme2 } from '@grafana/data';
import { selectors } from '@grafana/e2e-selectors'; import { selectors } from '@grafana/e2e-selectors';
import { reportInteraction } from '@grafana/runtime'; import { useStyles2 } from '@grafana/ui';
import { Themeable2, withTheme2 } from '@grafana/ui';
import { config } from 'app/core/config';
import store from 'app/core/store';
import {
DEFAULT_RANGE,
ensureQueries,
queryDatasourceDetails,
getTimeRange,
getTimeRangeFromUrl,
lastUsedDatasourceKeyForOrgId,
parseUrlState,
} from 'app/core/utils/explore';
import { MIXED_DATASOURCE_NAME } from 'app/plugins/datasource/mixed/MixedDataSource';
import { StoreState } from 'app/types'; import { StoreState } from 'app/types';
import { ExploreId } from 'app/types/explore'; import { ExploreId } from 'app/types/explore';
import { getDatasourceSrv } from '../plugins/datasource_srv';
import { getFiscalYearStartMonth, getTimeZone } from '../profile/state/selectors';
import Explore from './Explore'; import Explore from './Explore';
import { initializeExplore, refreshExplore } from './state/explorePane';
import { lastSavedUrl, stateSave } from './state/main';
import { importQueries } from './state/query';
import { loadAndInitDatasource } from './state/utils';
const getStyles = (theme: GrafanaTheme2) => { const getStyles = (theme: GrafanaTheme2) => {
return { return {
@@ -46,158 +25,42 @@ const getStyles = (theme: GrafanaTheme2) => {
}; };
}; };
interface OwnProps extends Themeable2 { interface Props {
exploreId: ExploreId; exploreId: ExploreId;
urlQuery: string;
eventBus: EventBus;
} }
interface Props extends OwnProps, ConnectedProps<typeof connector> {} /*
Connected components subscribe to the store before function components (using hooks) and can react to store changes. Thus, this connector function is called before the parent component (ExplorePage) is rerendered.
This means that child components' mapStateToProps will be executed with a zombie `exploreId` that is not present anymore in the store if the pane gets closed.
By connecting this component and returning the pane we workaround the zombie children issue here instead of modifying every children.
This is definitely not the ideal solution and we should in the future invest more time in exploring other approaches to better handle this scenario, potentially by refactoring panels to be function components
(therefore immune to this behaviour), or by forbidding them to access the store directly and instead pass them all the data they need via props or context.
/** You can read more about this issue here: https://react-redux.js.org/api/hooks#stale-props-and-zombie-children
* This component is responsible for handling initialization of an Explore pane and triggering synchronization
* of state based on URL changes and preventing any infinite loops.
*/ */
class ExplorePaneContainerUnconnected extends React.PureComponent<Props> { function ExplorePaneContainerUnconnected({ exploreId }: Props) {
el: HTMLDivElement | null = null; const styles = useStyles2(getStyles);
exploreEvents: EventBusExtended; const eventBus = useRef(new EventBusSrv());
const ref = useRef(null);
constructor(props: Props) { useEffect(() => {
super(props); const bus = eventBus.current;
this.exploreEvents = new EventBusSrv(); return () => bus.removeAllListeners();
this.state = { }, []);
openDrawer: undefined,
};
}
async componentDidMount() {
const {
initialized,
exploreId,
initialDatasource,
initialQueries,
initialRange,
panelsState,
orgId,
isFromCompactUrl,
} = this.props;
const width = this.el?.offsetWidth ?? 0;
// initialize the whole explore first time we mount and if browser history contains a change in datasource
if (!initialized) {
let queriesDatasourceOverride = undefined;
let rootDatasourceOverride = undefined;
// if this is starting with no queries and an initial datasource exists (but is not mixed), look up the ref to use it (initial datasource can be a UID or name here)
if ((!initialQueries || initialQueries.length === 0) && initialDatasource) {
const isDSMixed =
initialDatasource === MIXED_DATASOURCE_NAME || initialDatasource.uid === MIXED_DATASOURCE_NAME;
if (!isDSMixed) {
const { instance } = await loadAndInitDatasource(orgId, initialDatasource);
queriesDatasourceOverride = instance.getRef();
}
}
let queries = await ensureQueries(initialQueries, queriesDatasourceOverride); // this will return an empty array if there are no datasources
const queriesDatasourceDetails = queryDatasourceDetails(queries);
if (!queriesDatasourceDetails.noneHaveDatasource) {
if (!queryDatasourceDetails(queries).allDatasourceSame) {
if (config.featureToggles.exploreMixedDatasource) {
rootDatasourceOverride = await getDatasourceSrv().get(MIXED_DATASOURCE_NAME);
} else {
// if we have mixed queries but the mixed datasource feature is not on, change the datasource to the first query that has one
const changeDatasourceUid = queries.find((query) => query.datasource?.uid)!.datasource!.uid;
if (changeDatasourceUid) {
rootDatasourceOverride = changeDatasourceUid;
const datasource = await getDatasourceSrv().get(changeDatasourceUid);
const datasourceInit = await getDatasourceSrv().get(initialDatasource);
const newQueries = await this.props.importQueries(exploreId, queries, datasourceInit, datasource);
await this.props.stateSave({ replace: true });
queries = newQueries ?? this.props.initialQueries;
}
}
}
}
if (isFromCompactUrl) {
reportInteraction('grafana_explore_compact_notice');
}
this.props.initializeExplore({
exploreId,
datasource: rootDatasourceOverride || queries[0]?.datasource || initialDatasource,
queries,
range: initialRange,
containerWidth: width,
eventBridge: this.exploreEvents,
panelsState,
isFromCompactUrl,
});
}
}
componentWillUnmount() {
this.exploreEvents.removeAllListeners();
}
componentDidUpdate(prevProps: Props) {
this.refreshExplore(prevProps.urlQuery);
}
refreshExplore = (prevUrlQuery: string) => {
const { exploreId, urlQuery } = this.props;
// Update state from url only if it changed and only if the change wasn't initialised by redux to prevent any loops
if (urlQuery !== prevUrlQuery && urlQuery !== lastSavedUrl[exploreId]) {
this.props.refreshExplore(exploreId, urlQuery);
}
};
getRef = (el: HTMLDivElement) => {
this.el = el;
};
render() {
const { theme, exploreId, initialized, eventBus } = this.props;
const styles = getStyles(theme);
return ( return (
<div className={styles.explore} ref={this.getRef} data-testid={selectors.pages.Explore.General.container}> <div className={styles.explore} ref={ref} data-testid={selectors.pages.Explore.General.container}>
{initialized && <Explore exploreId={exploreId} eventBus={eventBus} />} <Explore exploreId={exploreId} eventBus={eventBus.current} />
</div> </div>
); );
} }
function mapStateToProps(state: StoreState, props: Props) {
const pane = state.explore.panes[props.exploreId];
return { pane };
} }
const getTimeRangeFromUrlMemoized = memoizeOne(getTimeRangeFromUrl); const connector = connect(mapStateToProps);
function mapStateToProps(state: StoreState, props: OwnProps) { export const ExplorePaneContainer = connector(ExplorePaneContainerUnconnected);
const urlState = parseUrlState(props.urlQuery);
const timeZone = getTimeZone(state.user);
const fiscalYearStartMonth = getFiscalYearStartMonth(state.user);
const { datasource, queries, range: urlRange, panelsState } = urlState || {};
const initialDatasource = datasource || store.get(lastUsedDatasourceKeyForOrgId(state.user.orgId));
const initialRange = urlRange
? getTimeRangeFromUrlMemoized(urlRange, timeZone, fiscalYearStartMonth)
: getTimeRange(timeZone, DEFAULT_RANGE, fiscalYearStartMonth);
return {
initialized: state.explore.panes[props.exploreId]?.initialized,
initialDatasource,
initialQueries: queries,
initialRange,
panelsState,
orgId: state.user.orgId,
isFromCompactUrl: urlState.isFromCompactUrl || false,
};
}
const mapDispatchToProps = {
initializeExplore,
refreshExplore,
importQueries,
stateSave,
};
const connector = connect(mapStateToProps, mapDispatchToProps);
export const ExplorePaneContainer = withTheme2(connector(ExplorePaneContainerUnconnected));

View File

@@ -68,7 +68,9 @@ export function ExploreQueryInspector(props: Props) {
label: 'Query', label: 'Query',
value: 'query', value: 'query',
icon: 'info-circle', icon: 'info-circle',
content: <QueryInspector data={dataFrames} onRefreshQuery={() => props.runQueries(props.exploreId)} />, content: (
<QueryInspector data={dataFrames} onRefreshQuery={() => props.runQueries({ exploreId: props.exploreId })} />
),
}; };
const tabs = [statsTab, queryTab, jsonTab, dataTab]; const tabs = [statsTab, queryTab, jsonTab, dataTab];

View File

@@ -88,7 +88,7 @@ export function ExploreToolbar({ exploreId, topOfViewRef, onChangeTime }: Props)
if (loading) { if (loading) {
return dispatch(cancelQueries(exploreId)); return dispatch(cancelQueries(exploreId));
} else { } else {
return dispatch(runQueries(exploreId)); return dispatch(runQueries({ exploreId }));
} }
}; };
@@ -119,8 +119,8 @@ export function ExploreToolbar({ exploreId, topOfViewRef, onChangeTime }: Props)
const onChangeFiscalYearStartMonth = (fiscalyearStartMonth: number) => const onChangeFiscalYearStartMonth = (fiscalyearStartMonth: number) =>
dispatch(updateFiscalYearStartMonthForSession(fiscalyearStartMonth)); dispatch(updateFiscalYearStartMonthForSession(fiscalyearStartMonth));
const onChangeRefreshInterval = (item: string) => { const onChangeRefreshInterval = (refreshInterval: string) => {
dispatch(changeRefreshInterval(exploreId, item)); dispatch(changeRefreshInterval({ exploreId, refreshInterval }));
}; };
const showExploreToDashboard = useMemo( const showExploreToDashboard = useMemo(

View File

@@ -11,7 +11,7 @@ import { ExploreId } from 'app/types/explore';
import { getDatasourceSrv } from '../plugins/datasource_srv'; import { getDatasourceSrv } from '../plugins/datasource_srv';
import { QueryEditorRows } from '../query/components/QueryEditorRows'; import { QueryEditorRows } from '../query/components/QueryEditorRows';
import { runQueries, changeQueries } from './state/query'; import { changeQueries, runQueries } from './state/query';
import { getExploreItemSelector } from './state/selectors'; import { getExploreItemSelector } from './state/selectors';
interface Props { interface Props {
@@ -39,14 +39,14 @@ export const QueryRows = ({ exploreId }: Props) => {
[exploreId] [exploreId]
); );
const queries = useSelector(getQueries)!; const queries = useSelector(getQueries);
const dsSettings = useSelector(getDatasourceInstanceSettings)!; const dsSettings = useSelector(getDatasourceInstanceSettings);
const queryResponse = useSelector(getQueryResponse)!; const queryResponse = useSelector(getQueryResponse);
const history = useSelector(getHistory); const history = useSelector(getHistory);
const eventBridge = useSelector(getEventBridge); const eventBridge = useSelector(getEventBridge);
const onRunQueries = useCallback(() => { const onRunQueries = useCallback(() => {
dispatch(runQueries(exploreId)); dispatch(runQueries({ exploreId }));
}, [dispatch, exploreId]); }, [dispatch, exploreId]);
const onChange = useCallback( const onChange = useCallback(

View File

@@ -1,14 +1,15 @@
import { render, screen } from '@testing-library/react'; import { render, screen } from '@testing-library/react';
import React from 'react'; import React from 'react';
import { Provider } from 'react-redux'; import { TestProvider } from 'test/helpers/TestProvider';
import { DataQueryError, LoadingState, getDefaultTimeRange } from '@grafana/data'; import { DataQueryError, LoadingState } from '@grafana/data';
import { selectors } from '@grafana/e2e-selectors'; import { selectors } from '@grafana/e2e-selectors';
import { ExploreId } from 'app/types'; import { ExploreId } from 'app/types';
import { configureStore } from '../../store/configureStore'; import { configureStore } from '../../store/configureStore';
import { ResponseErrorContainer } from './ResponseErrorContainer'; import { ResponseErrorContainer } from './ResponseErrorContainer';
import { createEmptyQueryResponse, makeExplorePaneState } from './state/utils';
describe('ResponseErrorContainer', () => { describe('ResponseErrorContainer', () => {
it('shows error message if it does not contain refId', async () => { it('shows error message if it does not contain refId', async () => {
@@ -46,26 +47,20 @@ describe('ResponseErrorContainer', () => {
function setup(error: DataQueryError) { function setup(error: DataQueryError) {
const store = configureStore(); const store = configureStore();
store.getState().explore.panes.left!.queryResponse = { store.getState().explore.panes = {
timeRange: getDefaultTimeRange(), left: {
series: [], ...makeExplorePaneState(),
queryResponse: {
...createEmptyQueryResponse(),
state: LoadingState.Error, state: LoadingState.Error,
error, error,
graphFrames: [], },
logsFrames: [], },
tableFrames: [],
traceFrames: [],
nodeGraphFrames: [],
rawPrometheusFrames: [],
flameGraphFrames: [],
graphResult: null,
logsResult: null,
tableResult: null,
rawPrometheusResult: null,
}; };
render( render(
<Provider store={store}> <TestProvider store={store}>
<ResponseErrorContainer exploreId={ExploreId.left} /> <ResponseErrorContainer exploreId={ExploreId.left} />
</Provider> </TestProvider>
); );
} }

View File

@@ -0,0 +1,35 @@
import { useEffect } from 'react';
import { config } from '@grafana/runtime';
import { useAppNotification } from 'app/core/copy/appNotification';
import { useCorrelations } from 'app/features/correlations/useCorrelations';
import { useDispatch } from 'app/types';
import { saveCorrelationsAction } from '../state/main';
export function useExploreCorrelations() {
const { get } = useCorrelations();
const { warning } = useAppNotification();
const dispatch = useDispatch();
useEffect(() => {
if (!config.featureToggles.correlations) {
dispatch(saveCorrelationsAction([]));
} else {
get.execute();
}
// eslint-disable-next-line react-hooks/exhaustive-deps
}, []);
useEffect(() => {
if (get.value) {
dispatch(saveCorrelationsAction(get.value));
} else if (get.error) {
dispatch(saveCorrelationsAction([]));
warning(
'Could not load correlations.',
'Correlations data could not be loaded, DataLinks may have partial data.'
);
}
}, [get.value, get.error, dispatch, warning]);
}

View File

@@ -0,0 +1,16 @@
import { isTruthy } from '@grafana/data';
import { Branding } from 'app/core/components/Branding/Branding';
import { useNavModel } from 'app/core/hooks/useNavModel';
import { useSelector } from 'app/types';
import { selectOrderedExplorePanes } from '../state/selectors';
export function useExplorePageTitle() {
const navModel = useNavModel('explore');
const datasourceNames = useSelector((state) =>
Object.values(selectOrderedExplorePanes(state)).map((pane) => pane?.datasourceInstance?.name)
).filter(isTruthy);
document.title = `${navModel.main.text} - ${datasourceNames.join(' | ')} - ${Branding.AppTitle}`;
}

View File

@@ -0,0 +1,409 @@
import { isEmpty, isEqual, isObject, mapValues, omitBy } from 'lodash';
import { useEffect, useRef } from 'react';
import {
CoreApp,
serializeStateToUrlParam,
ExploreUrlState,
isDateTime,
TimeRange,
RawTimeRange,
DataSourceApi,
} from '@grafana/data';
import { DataQuery, DataSourceRef } from '@grafana/schema';
import { useGrafana } from 'app/core/context/GrafanaContext';
import { clearQueryKeys, getLastUsedDatasourceUID, parseUrlState } from 'app/core/utils/explore';
import { getDatasourceSrv } from 'app/features/plugins/datasource_srv';
import { MIXED_DATASOURCE_NAME } from 'app/plugins/datasource/mixed/MixedDataSource';
import { addListener, ExploreId, ExploreItemState, ExploreQueryParams, useDispatch, useSelector } from 'app/types';
import { changeDatasource } from '../state/datasource';
import { initializeExplore } from '../state/explorePane';
import { clearPanes, splitClose, splitOpen, syncTimesAction } from '../state/main';
import { runQueries, setQueriesAction } from '../state/query';
import { selectPanes } from '../state/selectors';
import { changeRangeAction, updateTime } from '../state/time';
import { DEFAULT_RANGE } from '../state/utils';
import { withUniqueRefIds } from '../utils/queries';
/**
* Bi-directionally syncs URL changes with Explore's state.
*/
export function useStateSync(params: ExploreQueryParams) {
const {
location,
config: {
featureToggles: { exploreMixedDatasource },
},
} = useGrafana();
const dispatch = useDispatch();
const statePanes = useSelector(selectPanes);
const orgId = useSelector((state) => state.user.orgId);
const prevParams = useRef<ExploreQueryParams>(params);
const initState = useRef<'notstarted' | 'pending' | 'done'>('notstarted');
useEffect(() => {
// This happens when the user navigates to an explore "empty page" while within Explore.
// ie. by clicking on the explore when explore is active.
if (!params.left && !params.right) {
initState.current = 'notstarted';
prevParams.current = params;
}
}, [params]);
useEffect(() => {
const unsubscribe = dispatch(
addListener({
predicate: (action) =>
// We want to update the URL when:
// - a pane is opened or closed
// - a query is run
// - range is changed
[splitClose.type, splitOpen.fulfilled.type, runQueries.pending.type, changeRangeAction.type].includes(
action.type
),
effect: async (_, { cancelActiveListeners, delay, getState }) => {
// The following 2 lines will throttle updates to avoid creating history entries when rapid changes
// are committed to the store.
cancelActiveListeners();
await delay(200);
const panesQueryParams = Object.entries(getState().explore.panes).reduce<Record<string, string>>(
(acc, [id, paneState]) => ({ ...acc, [id]: serializeStateToUrlParam(getUrlStateFromPaneState(paneState)) }),
{}
);
if (!isEqual(prevParams.current, panesQueryParams)) {
// If there's no previous state it means we are mounting explore for the first time,
// in this case we want to replace the URL instead of pushing a new entry to the history.
const replace = Object.values(prevParams.current).filter(Boolean).length === 0;
prevParams.current = panesQueryParams;
location.partial(
{ left: panesQueryParams.left, right: panesQueryParams.right, orgId: getState().user.orgId },
replace
);
}
},
})
);
// @ts-expect-error the return type of addListener is actually callable, but dispatch is not middleware-aware
return () => unsubscribe();
}, [dispatch, location]);
useEffect(() => {
const isURLOutOfSync = prevParams.current?.left !== params.left || prevParams.current?.right !== params.right;
const urlPanes = {
left: parseUrlState(params.left),
...(params.right && { right: parseUrlState(params.right) }),
};
async function sync() {
// if navigating the history causes one of the time range to not being equal to all the other ones,
// we set syncedTimes to false to avoid inconsistent UI state.
// Ideally `syncedTimes` should be saved in the URL.
if (Object.values(urlPanes).some(({ range }, _, [{ range: firstRange }]) => !isEqual(range, firstRange))) {
dispatch(syncTimesAction({ syncedTimes: false }));
}
for (const [exploreId, urlPane] of Object.entries(urlPanes) as Array<[ExploreId, ExploreUrlState]>) {
const { datasource, queries, range, panelsState } = urlPane;
const statePane = statePanes[exploreId];
if (statePane !== undefined) {
// TODO: the diff contains panelState updates, but we are currently not handling them.
const update = urlDiff(urlPane, getUrlStateFromPaneState(statePane));
Promise.resolve()
.then(async () => {
if (update.datasource) {
await dispatch(changeDatasource(exploreId, datasource));
}
return;
})
.then(() => {
if (update.range) {
dispatch(updateTime({ exploreId, rawRange: range }));
}
if (update.queries) {
dispatch(setQueriesAction({ exploreId, queries: withUniqueRefIds(queries) }));
}
if (update.queries || update.range) {
dispatch(runQueries({ exploreId }));
}
});
} else {
// This happens when browser history is used to navigate.
// In this case we want to initialize the pane with the data from the URL
// if it's not present in the store. This may happen if the user has navigated
// from split view to non-split view and then back to split view.
dispatch(
initializeExplore({
exploreId,
datasource,
queries: withUniqueRefIds(queries),
range,
panelsState,
})
);
}
}
// Close all the panes that are not in the URL but are still in the store
// ie. because the user has navigated back after opening the split view.
Object.keys(statePanes)
.filter((keyInStore) => !Object.keys(urlPanes).includes(keyInStore))
.forEach((paneId) => dispatch(splitClose(paneId as ExploreId)));
}
// This happens when the user first navigates to explore.
// Here we want to initialize each pane initial data, wether it comes
// from the url or as a result of migrations.
if (!isURLOutOfSync && initState.current === 'notstarted') {
initState.current = 'pending';
// Clear all the panes in the store first to avoid stale data.
dispatch(clearPanes());
Promise.all(
Object.entries(urlPanes).map(([exploreId, { datasource, queries, range, panelsState }]) => {
return getPaneDatasource(datasource, queries, orgId, !!exploreMixedDatasource).then(
async (paneDatasource) => {
return Promise.resolve(
// In theory, given the Grafana datasource will always be present, this should always be defined.
paneDatasource
? queries.length
? // if we have queries in the URL, we use them
withUniqueRefIds(queries)
// but filter out the ones that are not compatible with the pane datasource
.filter(getQueryFilter(paneDatasource))
: getDatasourceSrv()
// otherwise we get a default query from the pane datasource or from the default datasource if the pane datasource is mixed
.get(isMixedDatasource(paneDatasource) ? undefined : paneDatasource.getRef())
.then((ds) => [getDefaultQuery(ds)])
: []
)
.then(async (queries) => {
// we remove queries that have an invalid datasources
const validQueries = await removeQueriesWithInvalidDatasource(queries);
if (!validQueries.length && paneDatasource) {
// and in case there's no query left we add a default one.
return [
getDefaultQuery(
isMixedDatasource(paneDatasource) ? await getDatasourceSrv().get() : paneDatasource
),
];
}
return validQueries;
})
.then((queries) => {
return dispatch(
initializeExplore({
exploreId: exploreId as ExploreId,
datasource: paneDatasource,
queries,
range,
panelsState,
})
).unwrap();
});
}
);
})
).then((panes) => {
const urlState = panes.reduce<ExploreQueryParams>((acc, { exploreId, state }) => {
return { ...acc, [exploreId]: serializeStateToUrlParam(getUrlStateFromPaneState(state)) };
}, {});
location.partial({ ...urlState, orgId }, true);
initState.current = 'done';
});
}
prevParams.current = {
left: params.left,
};
if (params.right) {
prevParams.current.right = params.right;
}
isURLOutOfSync && initState.current === 'done' && sync();
}, [params.left, params.right, dispatch, statePanes, exploreMixedDatasource, orgId, location]);
}
function getDefaultQuery(ds: DataSourceApi) {
return { ...ds.getDefaultQuery?.(CoreApp.Explore), refId: 'A', datasource: ds.getRef() };
}
function isMixedDatasource(datasource: DataSourceApi) {
return datasource.name === MIXED_DATASOURCE_NAME;
}
function getQueryFilter(datasource?: DataSourceApi) {
// if the root datasource is mixed, filter out queries that don't have a datasource.
if (datasource && isMixedDatasource(datasource)) {
return (q: DataQuery) => !!q.datasource;
} else {
// else filter out queries that have a datasource different from the root one.
// Queries may not have a datasource, if so, it's assumed they are using the root datasource
return (q: DataQuery) => {
if (!q.datasource) {
return true;
}
// Due to legacy URLs, `datasource` in queries may be a string. This logic should probably be in the migration
if (typeof q.datasource === 'string') {
return q.datasource === datasource?.uid;
}
return q.datasource.uid === datasource?.uid;
};
}
}
async function removeQueriesWithInvalidDatasource(queries: DataQuery[]) {
const results = await Promise.allSettled(
queries.map((query) => {
return getDatasourceSrv()
.get(query.datasource)
.then((ds) => ({
query,
ds,
}));
})
);
return results.filter(isFulfilled).map(({ value }) => value.query);
}
/**
* Returns the datasource that an explore pane should be using.
* If the URL specifies a datasource and that datasource exists, it will be used unless said datasource is mixed and `allowMixed` is false.
* Otherwise the datasource will be extracetd from the the first query specifying a valid datasource.
*
* If there's no datasource in the queries, the last used datasource will be used.
* if there's no last used datasource, the default datasource will be used.
*
* @param rootDatasource the top-level datasource specified in the URL
* @param queries the queries in the pane
* @param orgId the orgId of the user
* @param allowMixed whether mixed datasources are allowed
*
* @returns the datasource UID that the pane should use, undefined if no suitable datasource is found
*/
async function getPaneDatasource(
rootDatasource: DataSourceRef | string | null | undefined,
queries: DataQuery[],
orgId: number,
allowMixed: boolean
) {
// If there's a root datasource, use it unless it's mixed and we don't allow mixed.
if (rootDatasource) {
try {
const ds = await getDatasourceSrv().get(rootDatasource);
if (!isMixedDatasource(ds) || allowMixed) {
return ds;
}
} catch (_) {}
}
// TODO: if queries have multiple datasources and allowMixed is true, we should return mixed datasource
// Else we try to find a datasource in the queries, returning the first one that exists
const queriesWithDS = queries.filter((q) => q.datasource);
for (const query of queriesWithDS) {
try {
return await getDatasourceSrv().get(query.datasource);
} catch (_) {}
}
// If none of the queries specify a avalid datasource, we use the last used one
const lastUsedDSUID = getLastUsedDatasourceUID(orgId);
return (
getDatasourceSrv()
.get(lastUsedDSUID)
// Or the default one
.catch(() => getDatasourceSrv().get())
.catch(() => undefined)
);
}
const isFulfilled = <T>(promise: PromiseSettledResult<T>): promise is PromiseFulfilledResult<T> =>
promise.status === 'fulfilled';
/**
* Compare 2 explore urls and return a map of what changed. Used to update the local state with all the
* side effects needed.
*/
const urlDiff = (
oldUrlState: ExploreUrlState | undefined,
currentUrlState: ExploreUrlState | undefined
): {
datasource: boolean;
queries: boolean;
range: boolean;
panelsState: boolean;
} => {
const datasource = !isEqual(currentUrlState?.datasource, oldUrlState?.datasource);
const queries = !isEqual(currentUrlState?.queries, oldUrlState?.queries);
const range = !isEqual(currentUrlState?.range || DEFAULT_RANGE, oldUrlState?.range || DEFAULT_RANGE);
const panelsState = !isEqual(currentUrlState?.panelsState, oldUrlState?.panelsState);
return {
datasource,
queries,
range,
panelsState,
};
};
function getUrlStateFromPaneState(pane: ExploreItemState): ExploreUrlState {
return {
// datasourceInstance should not be undefined anymore here but in case there is some path for it to be undefined
// lets just fallback instead of crashing.
datasource: pane.datasourceInstance?.uid || '',
queries: pane.queries.map(clearQueryKeys),
range: toRawTimeRange(pane.range),
// don't include panelsState in the url unless a piece of state is actually set
panelsState: pruneObject(pane.panelsState),
};
}
/**
* recursively walks an object, removing keys where the value is undefined
* if the resulting object is empty, returns undefined
**/
function pruneObject(obj: object): object | undefined {
let pruned = mapValues(obj, (value) => (isObject(value) ? pruneObject(value) : value));
pruned = omitBy<typeof pruned>(pruned, isEmpty);
if (isEmpty(pruned)) {
return undefined;
}
return pruned;
}
export const toRawTimeRange = (range: TimeRange): RawTimeRange => {
let from = range.raw.from;
if (isDateTime(from)) {
from = from.valueOf().toString(10);
}
let to = range.raw.to;
if (isDateTime(to)) {
to = to.valueOf().toString(10);
}
return {
from,
to,
};
};

View File

@@ -0,0 +1,23 @@
import { useEffect, useRef } from 'react';
import { stopQueryState } from 'app/core/utils/explore';
import { useSelector } from 'app/types';
import { selectPanes } from '../state/selectors';
/**
* Unsubscribe from queries when unmounting.
* This avoids unnecessary state changes when navigating away from Explore.
*/
export function useStopQueries() {
const panesRef = useRef<ReturnType<typeof selectPanes>>({});
panesRef.current = useSelector(selectPanes);
useEffect(() => {
return () => {
for (const [, pane] of Object.entries(panesRef.current)) {
stopQueryState(pane.querySubscription);
}
};
}, []);
}

View File

@@ -0,0 +1,24 @@
import { useEffect } from 'react';
import { useGrafana } from 'app/core/context/GrafanaContext';
/**
* timeSrv (which is used internally) on init reads `from` and `to` param from the URL and updates itself
* using those value regardless of what is passed to the init method.
* The updated value is then used by Explore to get the range for each pane.
* This means that if `from` and `to` parameters are present in the URL,
* it would be impossible to change the time range in Explore.
* We are only doing this on mount for 2 reasons:
* 1: Doing it on update means we'll enter a render loop.
* 2: when parsing time in Explore (before feeding it to timeSrv) we make sure `from` is before `to` inside
* each pane state in order to not trigger un URL update from timeSrv.
*/
export function useTimeSrvFix() {
const { location } = useGrafana();
useEffect(() => {
const searchParams = location.getSearchObject();
if (searchParams.from || searchParams.to) {
location.partial({ from: undefined, to: undefined }, true);
}
}, [location]);
}

View File

@@ -1,7 +1,6 @@
import { screen, waitFor } from '@testing-library/react'; import { screen, waitFor } from '@testing-library/react';
import { serializeStateToUrlParam } from '@grafana/data'; import { serializeStateToUrlParam } from '@grafana/data';
import { locationService } from '@grafana/runtime';
import { changeDatasource } from './helper/interactions'; import { changeDatasource } from './helper/interactions';
import { makeLogsQueryResponse } from './helper/query'; import { makeLogsQueryResponse } from './helper/query';
@@ -18,14 +17,16 @@ describe('Explore: handle datasource states', () => {
it('handles changing the datasource manually', async () => { it('handles changing the datasource manually', async () => {
const urlParams = { left: JSON.stringify(['now-1h', 'now', 'loki', { expr: '{ label="value"}', refId: 'A' }]) }; const urlParams = { left: JSON.stringify(['now-1h', 'now', 'loki', { expr: '{ label="value"}', refId: 'A' }]) };
const { datasources } = setupExplore({ urlParams }); const { datasources, location } = setupExplore({ urlParams });
jest.mocked(datasources.loki.query).mockReturnValueOnce(makeLogsQueryResponse()); jest.mocked(datasources.loki.query).mockReturnValueOnce(makeLogsQueryResponse());
await waitForExplore(); await waitForExplore();
await changeDatasource('elastic'); await changeDatasource('elastic');
await screen.findByText('elastic Editor input:'); await screen.findByText('elastic Editor input:');
expect(datasources.elastic.query).not.toBeCalled(); expect(datasources.elastic.query).not.toBeCalled();
expect(locationService.getSearchObject()).toEqual({
await waitFor(async () => {
expect(location.getSearchObject()).toEqual({
orgId: '1', orgId: '1',
left: serializeStateToUrlParam({ left: serializeStateToUrlParam({
datasource: 'elastic-uid', datasource: 'elastic-uid',
@@ -35,3 +36,4 @@ describe('Explore: handle datasource states', () => {
}); });
}); });
}); });
});

View File

@@ -1,5 +1,8 @@
import { render, screen, within } from '@testing-library/react'; import { waitFor, within } from '@testing-library/dom';
import { render, screen } from '@testing-library/react';
import { createMemoryHistory } from 'history';
import { fromPairs } from 'lodash'; import { fromPairs } from 'lodash';
import { stringify } from 'querystring';
import React from 'react'; import React from 'react';
import { Provider } from 'react-redux'; import { Provider } from 'react-redux';
import { Route, Router } from 'react-router-dom'; import { Route, Router } from 'react-router-dom';
@@ -8,35 +11,40 @@ import { getGrafanaContextMock } from 'test/mocks/getGrafanaContextMock';
import { import {
DataSourceApi, DataSourceApi,
DataSourceInstanceSettings, DataSourceInstanceSettings,
DataSourceRef,
QueryEditorProps, QueryEditorProps,
ScopedVars, DataSourcePluginMeta,
UrlQueryValue, PluginType,
} from '@grafana/data'; } from '@grafana/data';
import { locationSearchToObject, locationService, setDataSourceSrv, setEchoSrv, config } from '@grafana/runtime'; import {
setDataSourceSrv,
setEchoSrv,
config,
setLocationService,
HistoryWrapper,
LocationService,
} from '@grafana/runtime';
import { DataSourceRef } from '@grafana/schema';
import { GrafanaContext } from 'app/core/context/GrafanaContext'; import { GrafanaContext } from 'app/core/context/GrafanaContext';
import { GrafanaRoute } from 'app/core/navigation/GrafanaRoute'; import { GrafanaRoute } from 'app/core/navigation/GrafanaRoute';
import { Echo } from 'app/core/services/echo/Echo'; import { Echo } from 'app/core/services/echo/Echo';
import store from 'app/core/store'; import { setLastUsedDatasourceUID } from 'app/core/utils/explore';
import { lastUsedDatasourceKeyForOrgId } from 'app/core/utils/explore';
import { MIXED_DATASOURCE_NAME } from 'app/plugins/datasource/mixed/MixedDataSource'; import { MIXED_DATASOURCE_NAME } from 'app/plugins/datasource/mixed/MixedDataSource';
import { configureStore } from 'app/store/configureStore'; import { configureStore } from 'app/store/configureStore';
import { LokiDatasource } from '../../../../plugins/datasource/loki/datasource'; import { LokiDatasource } from '../../../../plugins/datasource/loki/datasource';
import { LokiQuery } from '../../../../plugins/datasource/loki/types'; import { LokiQuery } from '../../../../plugins/datasource/loki/types';
import { ExploreId } from '../../../../types'; import { ExploreId, ExploreQueryParams } from '../../../../types';
import { initialUserState } from '../../../profile/state/reducers'; import { initialUserState } from '../../../profile/state/reducers';
import ExplorePage from '../../ExplorePage'; import ExplorePage from '../../ExplorePage';
type DatasourceSetup = { settings: DataSourceInstanceSettings; api: DataSourceApi }; type DatasourceSetup = { settings: DataSourceInstanceSettings; api: DataSourceApi };
type SetupOptions = { type SetupOptions = {
// default true
clearLocalStorage?: boolean; clearLocalStorage?: boolean;
datasources?: DatasourceSetup[]; datasources?: DatasourceSetup[];
urlParams?: { left: string; right?: string } | string; urlParams?: ExploreQueryParams & { [key: string]: string };
searchParams?: string;
prevUsedDatasource?: { orgId: number; datasource: string }; prevUsedDatasource?: { orgId: number; datasource: string };
mixedEnabled?: boolean;
}; };
export function setupExplore(options?: SetupOptions): { export function setupExplore(options?: SetupOptions): {
@@ -44,6 +52,7 @@ export function setupExplore(options?: SetupOptions): {
store: ReturnType<typeof configureStore>; store: ReturnType<typeof configureStore>;
unmount: () => void; unmount: () => void;
container: HTMLElement; container: HTMLElement;
location: LocationService;
} { } {
// Clear this up otherwise it persists data source selection // Clear this up otherwise it persists data source selection
// TODO: probably add test for that too // TODO: probably add test for that too
@@ -52,7 +61,7 @@ export function setupExplore(options?: SetupOptions): {
} }
if (options?.prevUsedDatasource) { if (options?.prevUsedDatasource) {
store.set(lastUsedDatasourceKeyForOrgId(options?.prevUsedDatasource.orgId), options?.prevUsedDatasource.datasource); setLastUsedDatasourceUID(options?.prevUsedDatasource.orgId, options?.prevUsedDatasource.datasource);
} }
// Create this here so any mocks are recreated on setup and don't retain state // Create this here so any mocks are recreated on setup and don't retain state
@@ -62,7 +71,7 @@ export function setupExplore(options?: SetupOptions): {
]; ];
if (config.featureToggles.exploreMixedDatasource) { if (config.featureToggles.exploreMixedDatasource) {
defaultDatasources.push(makeDatasourceSetup({ name: MIXED_DATASOURCE_NAME, id: 999 })); defaultDatasources.push(makeDatasourceSetup({ name: MIXED_DATASOURCE_NAME, uid: MIXED_DATASOURCE_NAME, id: 999 }));
} }
const dsSettings = options?.datasources || defaultDatasources; const dsSettings = options?.datasources || defaultDatasources;
@@ -71,24 +80,30 @@ export function setupExplore(options?: SetupOptions): {
getList(): DataSourceInstanceSettings[] { getList(): DataSourceInstanceSettings[] {
return dsSettings.map((d) => d.settings); return dsSettings.map((d) => d.settings);
}, },
getInstanceSettings(ref: DataSourceRef) { getInstanceSettings(ref?: DataSourceRef) {
return dsSettings.map((d) => d.settings).find((x) => x.name === ref || x.uid === ref || x.uid === ref.uid); const allSettings = dsSettings.map((d) => d.settings);
return allSettings.find((x) => x.name === ref || x.uid === ref || x.uid === ref?.uid) || allSettings[0];
}, },
get(datasource?: string | DataSourceRef | null, scopedVars?: ScopedVars): Promise<DataSourceApi | undefined> { get(datasource?: string | DataSourceRef | null): Promise<DataSourceApi> {
if (dsSettings.length === 0) { let ds: DataSourceApi | undefined;
return Promise.resolve(undefined); if (!datasource) {
ds = dsSettings[0]?.api;
} else { } else {
const datasourceStr = typeof datasource === 'string'; ds = dsSettings.find((ds) =>
return Promise.resolve( typeof datasource === 'string'
(datasource ? ds.api.name === datasource || ds.api.uid === datasource
? dsSettings.find((d) => : ds.api.uid === datasource?.uid
datasourceStr ? d.api.name === datasource || d.api.uid === datasource : d.api.uid === datasource?.uid )?.api;
)
: dsSettings[0])!.api
);
} }
if (ds) {
return Promise.resolve(ds);
}
return Promise.reject();
}, },
} as any); reload() {},
});
setEchoSrv(new Echo()); setEchoSrv(new Echo());
@@ -109,43 +124,82 @@ export function setupExplore(options?: SetupOptions): {
}, },
}; };
locationService.push({ pathname: '/explore', search: options?.searchParams }); const history = createMemoryHistory({
initialEntries: [{ pathname: '/explore', search: stringify(options?.urlParams) }],
});
if (options?.urlParams) { const location = new HistoryWrapper(history);
let urlParams: Record<string, string | UrlQueryValue> = setLocationService(location);
typeof options.urlParams === 'string' ? locationSearchToObject(options.urlParams) : options.urlParams;
locationService.partial(urlParams);
}
const route = { component: ExplorePage }; const contextMock = getGrafanaContextMock({ location });
const { unmount, container } = render( const { unmount, container } = render(
<Provider store={storeState}> <Provider store={storeState}>
<GrafanaContext.Provider value={getGrafanaContextMock()}> <GrafanaContext.Provider
<Router history={locationService.getHistory()}> value={{
<Route path="/explore" exact render={(props) => <GrafanaRoute {...props} route={route as any} />} /> ...contextMock,
config: {
...contextMock.config,
featureToggles: {
exploreMixedDatasource: options?.mixedEnabled ?? false,
},
},
}}
>
<Router history={history}>
<Route
path="/explore"
exact
render={(props) => <GrafanaRoute {...props} route={{ component: ExplorePage, path: '/explore' }} />}
/>
</Router> </Router>
</GrafanaContext.Provider> </GrafanaContext.Provider>
</Provider> </Provider>
); );
return { datasources: fromPairs(dsSettings.map((d) => [d.api.name, d.api])), store: storeState, unmount, container }; return {
datasources: fromPairs(dsSettings.map((d) => [d.api.name, d.api])),
store: storeState,
unmount,
container,
location,
};
} }
function makeDatasourceSetup({ name = 'loki', id = 1 }: { name?: string; id?: number } = {}): DatasourceSetup { function makeDatasourceSetup({
const meta: any = { name = 'loki',
id = 1,
uid: uidOverride,
}: { name?: string; id?: number; uid?: string } = {}): DatasourceSetup {
const uid = uidOverride || `${name}-uid`;
const type = 'logs';
const meta: DataSourcePluginMeta = {
info: { info: {
author: {
name: 'Grafana',
},
description: '',
links: [],
screenshots: [],
updated: '',
version: '',
logos: { logos: {
small: '', small: '',
large: '',
}, },
}, },
id: id.toString(), id: id.toString(),
module: 'loki',
name,
type: PluginType.datasource,
baseUrl: '',
}; };
return { return {
settings: { settings: {
id, id,
uid: `${name}-uid`, uid,
type: 'logs', type,
name, name,
meta, meta,
access: 'proxy', access: 'proxy',
@@ -170,20 +224,19 @@ function makeDatasourceSetup({ name = 'loki', id = 1 }: { name?: string; id?: nu
}, },
}, },
name: name, name: name,
uid: `${name}-uid`, uid: uid,
query: jest.fn(), query: jest.fn(),
getRef: jest.fn().mockReturnValue({ type: 'logs', uid: `${name}-uid` }), getRef: () => ({ type, uid }),
meta, meta,
} as any, } as any,
}; };
} }
export const waitForExplore = async (exploreId: ExploreId = ExploreId.left, multi = false) => { export const waitForExplore = (exploreId: ExploreId = ExploreId.left) => {
if (multi) { return waitFor(async () => {
return await withinExplore(exploreId).findAllByText(/Editor/i); const container = screen.getAllByTestId('data-testid Explore');
} else { return within(container[exploreId === ExploreId.left ? 0 : 1]);
return await withinExplore(exploreId).findByText(/Editor/i); });
}
}; };
export const tearDown = () => { export const tearDown = () => {

View File

@@ -47,6 +47,7 @@ jest.mock('@grafana/runtime', () => ({
jest.mock('app/core/core', () => ({ jest.mock('app/core/core', () => ({
contextSrv: { contextSrv: {
hasPermission: () => true,
hasAccess: () => true, hasAccess: () => true,
isSignedIn: true, isSignedIn: true,
}, },
@@ -92,7 +93,7 @@ describe('Explore: Query History', () => {
it('adds new query history items after the query is run.', async () => { it('adds new query history items after the query is run.', async () => {
// when Explore is opened // when Explore is opened
const { datasources, unmount } = setupExplore(); const { datasources, unmount } = setupExplore();
(datasources.loki.query as jest.Mock).mockReturnValueOnce(makeLogsQueryResponse()); jest.mocked(datasources.loki.query).mockReturnValueOnce(makeLogsQueryResponse());
await waitForExplore(); await waitForExplore();
// and a user runs a query and opens query history // and a user runs a query and opens query history
@@ -127,7 +128,7 @@ describe('Explore: Query History', () => {
}; };
const { datasources } = setupExplore({ urlParams }); const { datasources } = setupExplore({ urlParams });
(datasources.loki.query as jest.Mock).mockReturnValueOnce(makeLogsQueryResponse()); jest.mocked(datasources.loki.query).mockReturnValueOnce(makeLogsQueryResponse());
await waitForExplore(); await waitForExplore();
await openQueryHistory(); await openQueryHistory();
@@ -136,7 +137,7 @@ describe('Explore: Query History', () => {
await assertQueryHistory(['{"expr":"query #2"}', '{"expr":"query #1"}']); await assertQueryHistory(['{"expr":"query #2"}', '{"expr":"query #1"}']);
}); });
it.skip('updates the state in both Explore panes', async () => { it('updates the state in both Explore panes', async () => {
const urlParams = { const urlParams = {
left: serializeStateToUrlParam({ left: serializeStateToUrlParam({
datasource: 'loki', datasource: 'loki',
@@ -151,7 +152,7 @@ describe('Explore: Query History', () => {
}; };
const { datasources } = setupExplore({ urlParams }); const { datasources } = setupExplore({ urlParams });
(datasources.loki.query as jest.Mock).mockReturnValue(makeLogsQueryResponse()); jest.mocked(datasources.loki.query).mockReturnValue(makeLogsQueryResponse());
await waitForExplore(); await waitForExplore();
await waitForExplore(ExploreId.right); await waitForExplore(ExploreId.right);
@@ -188,7 +189,7 @@ describe('Explore: Query History', () => {
}; };
const { datasources } = setupExplore({ urlParams }); const { datasources } = setupExplore({ urlParams });
(datasources.loki.query as jest.Mock).mockReturnValueOnce(makeLogsQueryResponse()); jest.mocked(datasources.loki.query).mockReturnValueOnce(makeLogsQueryResponse());
await waitForExplore(); await waitForExplore();
await openQueryHistory(); await openQueryHistory();
await assertQueryHistory(['{"expr":"query #1"}'], ExploreId.left); await assertQueryHistory(['{"expr":"query #1"}'], ExploreId.left);
@@ -222,7 +223,7 @@ describe('Explore: Query History', () => {
it('pagination', async () => { it('pagination', async () => {
config.queryHistoryEnabled = true; config.queryHistoryEnabled = true;
const { datasources } = setupExplore(); const { datasources } = setupExplore();
(datasources.loki.query as jest.Mock).mockReturnValueOnce(makeLogsQueryResponse()); jest.mocked(datasources.loki.query).mockReturnValueOnce(makeLogsQueryResponse());
fetchMock.mockReturnValue( fetchMock.mockReturnValue(
of({ of({
data: { result: { queryHistory: [{ datasourceUid: 'loki', queries: [{ expr: 'query' }] }], totalCount: 2 } }, data: { result: { queryHistory: [{ datasourceUid: 'loki', queries: [{ expr: 'query' }] }], totalCount: 2 } },

View File

@@ -3,6 +3,7 @@ import { AnyAction, createAction } from '@reduxjs/toolkit';
import { DataSourceApi, HistoryItem } from '@grafana/data'; import { DataSourceApi, HistoryItem } from '@grafana/data';
import { reportInteraction } from '@grafana/runtime'; import { reportInteraction } from '@grafana/runtime';
import { DataSourceRef } from '@grafana/schema';
import { RefreshPicker } from '@grafana/ui'; import { RefreshPicker } from '@grafana/ui';
import { stopQueryState } from 'app/core/utils/explore'; import { stopQueryState } from 'app/core/utils/explore';
import { ExploreItemState, ThunkResult } from 'app/types'; import { ExploreItemState, ThunkResult } from 'app/types';
@@ -39,12 +40,12 @@ export const updateDatasourceInstanceAction = createAction<UpdateDatasourceInsta
*/ */
export function changeDatasource( export function changeDatasource(
exploreId: ExploreId, exploreId: ExploreId,
datasourceUid: string, datasource: string | DataSourceRef,
options?: { importQueries: boolean } options?: { importQueries: boolean }
): ThunkResult<Promise<void>> { ): ThunkResult<Promise<void>> {
return async (dispatch, getState) => { return async (dispatch, getState) => {
const orgId = getState().user.orgId; const orgId = getState().user.orgId;
const { history, instance } = await loadAndInitDatasource(orgId, { uid: datasourceUid }); const { history, instance } = await loadAndInitDatasource(orgId, datasource);
const currentDataSourceInstance = getState().explore.panes[exploreId]!.datasourceInstance; const currentDataSourceInstance = getState().explore.panes[exploreId]!.datasourceInstance;
reportInteraction('explore_change_ds', { reportInteraction('explore_change_ds', {
@@ -66,12 +67,12 @@ export function changeDatasource(
} }
if (getState().explore.panes[exploreId]!.isLive) { if (getState().explore.panes[exploreId]!.isLive) {
dispatch(changeRefreshInterval(exploreId, RefreshPicker.offOption.value)); dispatch(changeRefreshInterval({ exploreId, refreshInterval: RefreshPicker.offOption.value }));
} }
// Exception - we only want to run queries on data source change, if the queries were imported // Exception - we only want to run queries on data source change, if the queries were imported
if (options?.importQueries) { if (options?.importQueries) {
dispatch(runQueries(exploreId)); dispatch(runQueries({ exploreId }));
} }
}; };
} }
@@ -106,7 +107,6 @@ export const datasourceReducer = (state: ExploreItemState, action: AnyAction): E
loading: false, loading: false,
queryKeys: [], queryKeys: [],
history, history,
datasourceMissing: false,
}; };
} }

View File

@@ -1,135 +0,0 @@
import { of } from 'rxjs';
import { serializeStateToUrlParam } from '@grafana/data';
import { setDataSourceSrv } from '@grafana/runtime';
import { ExploreId, StoreState, ThunkDispatch } from 'app/types';
import { configureStore } from '../../../store/configureStore';
import { refreshExplore } from './explorePane';
import { createDefaultInitialState } from './helpers';
jest.mock('../../dashboard/services/TimeSrv', () => ({
getTimeSrv: jest.fn().mockReturnValue({
init: jest.fn(),
timeRange: jest.fn().mockReturnValue({}),
}),
}));
const { testRange, defaultInitialState } = createDefaultInitialState();
jest.mock('@grafana/runtime', () => ({
...jest.requireActual('@grafana/runtime'),
getTemplateSrv: () => ({
updateTimeRange: jest.fn(),
}),
}));
function setupStore(state?: any) {
return configureStore({
...defaultInitialState,
explore: {
panes: {
[ExploreId.left]: {
...defaultInitialState.explore.panes.left,
...(state || {}),
},
},
},
} as any);
}
function setup(state?: any) {
const datasources: Record<string, any> = {
newDs: {
testDatasource: jest.fn(),
init: jest.fn(),
query: jest.fn(),
name: 'newDs',
meta: { id: 'newDs' },
getRef: () => ({ uid: 'newDs' }),
},
someDs: {
testDatasource: jest.fn(),
init: jest.fn(),
query: jest.fn(),
name: 'someDs',
meta: { id: 'someDs' },
getRef: () => ({ uid: 'someDs' }),
},
};
setDataSourceSrv({
getList() {
return Object.values(datasources).map((d) => ({ name: d.name }));
},
getInstanceSettings(name: string) {
return { name, getRef: () => ({ uid: name }) };
},
get(name?: string) {
return Promise.resolve(
name
? datasources[name]
: {
testDatasource: jest.fn(),
init: jest.fn(),
name: 'default',
getRef() {
return { type: 'default', uid: 'default' };
},
}
);
},
} as any);
const { dispatch, getState }: { dispatch: ThunkDispatch; getState: () => StoreState } = setupStore({
datasourceInstance: datasources.someDs,
...(state || {}),
});
return {
dispatch,
getState,
datasources,
};
}
describe('refreshExplore', () => {
it('should change data source when datasource in url changes', async () => {
const { dispatch, getState } = setup();
await dispatch(
refreshExplore(ExploreId.left, serializeStateToUrlParam({ datasource: 'newDs', queries: [], range: testRange }))
);
expect(getState().explore.panes.left!.datasourceInstance?.name).toBe('newDs');
});
it('should change and run new queries from the URL', async () => {
const { dispatch, getState, datasources } = setup();
datasources.someDs.query.mockReturnValueOnce(of({}));
await dispatch(
refreshExplore(
ExploreId.left,
serializeStateToUrlParam({ datasource: 'someDs', queries: [{ expr: 'count()', refId: 'A' }], range: testRange })
)
);
// same
const state = getState().explore.panes.left!;
expect(state.datasourceInstance?.name).toBe('someDs');
expect(state.queries.length).toBe(1);
expect(state.queries).toMatchObject([{ expr: 'count()' }]);
expect(datasources.someDs.query).toHaveBeenCalledTimes(1);
});
it('should not do anything if pane is not present', async () => {
const { dispatch, getState } = setup({});
const state = getState();
await dispatch(
refreshExplore(
ExploreId.right,
serializeStateToUrlParam({ datasource: 'newDs', queries: [{ expr: 'count()', refId: 'A' }], range: testRange })
)
);
expect(state).toEqual(getState());
});
});

View File

@@ -1,41 +1,26 @@
import { createAction, PayloadAction } from '@reduxjs/toolkit'; import { createAction, PayloadAction } from '@reduxjs/toolkit';
import { isEqual } from 'lodash';
import { AnyAction } from 'redux'; import { AnyAction } from 'redux';
import { import {
EventBusExtended,
ExploreUrlState,
TimeRange, TimeRange,
HistoryItem, HistoryItem,
DataSourceApi, DataSourceApi,
ExplorePanelsState, ExplorePanelsState,
PreferredVisualisationType, PreferredVisualisationType,
RawTimeRange,
} from '@grafana/data'; } from '@grafana/data';
import { getDataSourceSrv } from '@grafana/runtime';
import { DataQuery, DataSourceRef } from '@grafana/schema'; import { DataQuery, DataSourceRef } from '@grafana/schema';
import { import { getQueryKeys } from 'app/core/utils/explore';
DEFAULT_RANGE, import { getTimeZone } from 'app/features/profile/state/selectors';
getQueryKeys,
parseUrlState,
ensureQueries,
generateNewKeyAndAddRefIdIfMissing,
getTimeRangeFromUrl,
} from 'app/core/utils/explore';
import { getFiscalYearStartMonth, getTimeZone } from 'app/features/profile/state/selectors';
import { createAsyncThunk, ThunkResult } from 'app/types'; import { createAsyncThunk, ThunkResult } from 'app/types';
import { ExploreId, ExploreItemState } from 'app/types/explore'; import { ExploreId, ExploreItemState } from 'app/types/explore';
import { datasourceReducer } from './datasource'; import { datasourceReducer } from './datasource';
import { historyReducer } from './history'; import { historyReducer } from './history';
import { richHistorySearchFiltersUpdatedAction, richHistoryUpdatedAction, stateSave } from './main'; import { richHistorySearchFiltersUpdatedAction, richHistoryUpdatedAction } from './main';
import { queryReducer, runQueries, setQueriesAction } from './query'; import { queryReducer, runQueries } from './query';
import { timeReducer, updateTime } from './time'; import { timeReducer, updateTime } from './time';
import { import { makeExplorePaneState, loadAndInitDatasource, createEmptyQueryResponse, getRange } from './utils';
makeExplorePaneState,
loadAndInitDatasource,
createEmptyQueryResponse,
getUrlStateFromPaneState,
} from './utils';
// Types // Types
// //
@@ -49,7 +34,6 @@ import {
export interface ChangeSizePayload { export interface ChangeSizePayload {
exploreId: ExploreId; exploreId: ExploreId;
width: number; width: number;
height: number;
} }
export const changeSizeAction = createAction<ChangeSizePayload>('explore/changeSize'); export const changeSizeAction = createAction<ChangeSizePayload>('explore/changeSize');
@@ -81,7 +65,6 @@ export function changePanelState(
}, },
}) })
); );
dispatch(stateSave());
}; };
} }
@@ -91,13 +74,10 @@ export function changePanelState(
*/ */
interface InitializeExplorePayload { interface InitializeExplorePayload {
exploreId: ExploreId; exploreId: ExploreId;
containerWidth: number;
eventBridge: EventBusExtended;
queries: DataQuery[]; queries: DataQuery[];
range: TimeRange; range: TimeRange;
history: HistoryItem[]; history: HistoryItem[];
datasourceInstance?: DataSourceApi; datasourceInstance?: DataSourceApi;
isFromCompactUrl?: boolean;
} }
const initializeExploreAction = createAction<InitializeExplorePayload>('explore/initializeExploreAction'); const initializeExploreAction = createAction<InitializeExplorePayload>('explore/initializeExploreAction');
@@ -110,22 +90,16 @@ export const setUrlReplacedAction = createAction<SetUrlReplacedPayload>('explore
* Keep track of the Explore container size, in particular the width. * Keep track of the Explore container size, in particular the width.
* The width will be used to calculate graph intervals (number of datapoints). * The width will be used to calculate graph intervals (number of datapoints).
*/ */
export function changeSize( export function changeSize(exploreId: ExploreId, { width }: { width: number }): PayloadAction<ChangeSizePayload> {
exploreId: ExploreId, return changeSizeAction({ exploreId, width });
{ height, width }: { height: number; width: number }
): PayloadAction<ChangeSizePayload> {
return changeSizeAction({ exploreId, height, width });
} }
interface InitializeExploreOptions { interface InitializeExploreOptions {
exploreId: ExploreId; exploreId: ExploreId;
datasource: DataSourceRef | string; datasource: DataSourceRef | string | undefined;
queries: DataQuery[]; queries: DataQuery[];
range: TimeRange; range: RawTimeRange;
containerWidth: number;
eventBridge: EventBusExtended;
panelsState?: ExplorePanelsState; panelsState?: ExplorePanelsState;
isFromCompactUrl?: boolean;
} }
/** /**
* Initialize Explore state with state from the URL and the React component. * Initialize Explore state with state from the URL and the React component.
@@ -138,23 +112,13 @@ interface InitializeExploreOptions {
export const initializeExplore = createAsyncThunk( export const initializeExplore = createAsyncThunk(
'explore/initializeExplore', 'explore/initializeExplore',
async ( async (
{ { exploreId, datasource, queries, range, panelsState }: InitializeExploreOptions,
exploreId, { dispatch, getState, fulfillWithValue }
datasource,
queries,
range,
containerWidth,
eventBridge,
panelsState,
isFromCompactUrl,
}: InitializeExploreOptions,
{ dispatch, getState }
) => { ) => {
const exploreDatasources = getDataSourceSrv().getList();
let instance = undefined; let instance = undefined;
let history: HistoryItem[] = []; let history: HistoryItem[] = [];
if (exploreDatasources.length >= 1) { if (datasource) {
const orgId = getState().user.orgId; const orgId = getState().user.orgId;
const loadResult = await loadAndInitDatasource(orgId, datasource); const loadResult = await loadAndInitDatasource(orgId, datasource);
instance = loadResult.instance; instance = loadResult.instance;
@@ -164,13 +128,10 @@ export const initializeExplore = createAsyncThunk(
dispatch( dispatch(
initializeExploreAction({ initializeExploreAction({
exploreId, exploreId,
containerWidth,
eventBridge,
queries, queries,
range, range: getRange(range, getTimeZone(getState().user)),
datasourceInstance: instance, datasourceInstance: instance,
history, history,
isFromCompactUrl,
}) })
); );
if (panelsState !== undefined) { if (panelsState !== undefined) {
@@ -179,81 +140,13 @@ export const initializeExplore = createAsyncThunk(
dispatch(updateTime({ exploreId })); dispatch(updateTime({ exploreId }));
if (instance) { if (instance) {
// We do not want to add the url to browser history on init because when the pane is initialised it's because dispatch(runQueries({ exploreId }));
// we already have something in the url. Adding basically the same state as additional history item prevents
// user to go back to previous url.
dispatch(runQueries(exploreId, { replaceUrl: true }));
} }
return fulfillWithValue({ exploreId, state: getState().explore.panes[exploreId]! });
} }
); );
/**
* Reacts to changes in URL state that we need to sync back to our redux state. Computes diff of newUrlQuery vs current
* state and runs update actions for relevant parts.
*/
export function refreshExplore(exploreId: ExploreId, newUrlQuery: string): ThunkResult<void> {
return async (dispatch, getState) => {
const itemState = getState().explore.panes[exploreId];
if (!itemState) {
return;
}
// Get diff of what should be updated
const newUrlState = parseUrlState(newUrlQuery);
const update = urlDiff(newUrlState, getUrlStateFromPaneState(itemState));
const { containerWidth, eventBridge } = itemState;
// datasource will either be name or UID here
const { datasource, queries, range: urlRange, panelsState } = newUrlState;
const refreshQueries: DataQuery[] = [];
for (let index = 0; index < queries.length; index++) {
const query = queries[index];
refreshQueries.push(generateNewKeyAndAddRefIdIfMissing(query, refreshQueries, index));
}
const timeZone = getTimeZone(getState().user);
const fiscalYearStartMonth = getFiscalYearStartMonth(getState().user);
const range = getTimeRangeFromUrl(urlRange, timeZone, fiscalYearStartMonth);
// commit changes based on the diff of new url vs old url
if (update.datasource) {
const initialQueries = await ensureQueries(queries);
await dispatch(
initializeExplore({
exploreId,
datasource,
queries: initialQueries,
range,
containerWidth,
eventBridge,
panelsState,
})
);
return;
}
if (update.range) {
dispatch(updateTime({ exploreId, rawRange: range.raw }));
}
if (update.queries) {
dispatch(setQueriesAction({ exploreId, queries: refreshQueries }));
}
if (update.panelsState && panelsState !== undefined) {
dispatch(changePanelsStateAction({ exploreId, panelsState }));
}
// always run queries when refresh is needed
if (update.queries || update.range) {
dispatch(runQueries(exploreId));
}
};
}
/** /**
* Reducer for an Explore area, to be used by the global Explore reducer. * Reducer for an Explore area, to be used by the global Explore reducer.
*/ */
@@ -296,51 +189,20 @@ export const paneReducer = (state: ExploreItemState = makeExplorePaneState(), ac
} }
if (initializeExploreAction.match(action)) { if (initializeExploreAction.match(action)) {
const { containerWidth, eventBridge, queries, range, datasourceInstance, history, isFromCompactUrl } = const { queries, range, datasourceInstance, history } = action.payload;
action.payload;
return { return {
...state, ...state,
containerWidth,
eventBridge,
range, range,
queries, queries,
initialized: true, initialized: true,
queryKeys: getQueryKeys(queries), queryKeys: getQueryKeys(queries),
datasourceInstance, datasourceInstance,
history, history,
datasourceMissing: !datasourceInstance,
queryResponse: createEmptyQueryResponse(), queryResponse: createEmptyQueryResponse(),
cache: [], cache: [],
isFromCompactUrl: isFromCompactUrl || false,
}; };
} }
return state; return state;
}; };
/**
* Compare 2 explore urls and return a map of what changed. Used to update the local state with all the
* side effects needed.
*/
export const urlDiff = (
oldUrlState: ExploreUrlState | undefined,
currentUrlState: ExploreUrlState | undefined
): {
datasource: boolean;
queries: boolean;
range: boolean;
panelsState: boolean;
} => {
const datasource = !isEqual(currentUrlState?.datasource, oldUrlState?.datasource);
const queries = !isEqual(currentUrlState?.queries, oldUrlState?.queries);
const range = !isEqual(currentUrlState?.range || DEFAULT_RANGE, oldUrlState?.range || DEFAULT_RANGE);
const panelsState = !isEqual(currentUrlState?.panelsState, oldUrlState?.panelsState);
return {
datasource,
queries,
range,
panelsState,
};
};

View File

@@ -9,7 +9,7 @@ import { reducerTester } from '../../../../test/core/redux/reducerTester';
import { MockDataSourceApi } from '../../../../test/mocks/datasource_srv'; import { MockDataSourceApi } from '../../../../test/mocks/datasource_srv';
import { ExploreId, ExploreItemState, ExploreState } from '../../../types'; import { ExploreId, ExploreItemState, ExploreState } from '../../../types';
import { exploreReducer, navigateToExplore, splitCloseAction } from './main'; import { exploreReducer, navigateToExplore, splitClose } from './main';
const getNavigateToExploreContext = async (openInNewWindow?: (url: string) => void) => { const getNavigateToExploreContext = async (openInNewWindow?: (url: string) => void) => {
const url = '/explore'; const url = '/explore';
@@ -136,7 +136,7 @@ describe('Explore reducer', () => {
// closing left item // closing left item
reducerTester<ExploreState>() reducerTester<ExploreState>()
.givenReducer(exploreReducer, initialState) .givenReducer(exploreReducer, initialState)
.whenActionIsDispatched(splitCloseAction({ itemId: ExploreId.left })) .whenActionIsDispatched(splitClose(ExploreId.left))
.thenStateShouldEqual({ .thenStateShouldEqual({
evenSplitPanes: true, evenSplitPanes: true,
largerExploreId: undefined, largerExploreId: undefined,
@@ -166,7 +166,7 @@ describe('Explore reducer', () => {
// closing left item // closing left item
reducerTester<ExploreState>() reducerTester<ExploreState>()
.givenReducer(exploreReducer, initialState) .givenReducer(exploreReducer, initialState)
.whenActionIsDispatched(splitCloseAction({ itemId: ExploreId.right })) .whenActionIsDispatched(splitClose(ExploreId.right))
.thenStateShouldEqual({ .thenStateShouldEqual({
evenSplitPanes: true, evenSplitPanes: true,
largerExploreId: undefined, largerExploreId: undefined,
@@ -193,7 +193,7 @@ describe('Explore reducer', () => {
reducerTester<ExploreState>() reducerTester<ExploreState>()
.givenReducer(exploreReducer, initialState) .givenReducer(exploreReducer, initialState)
.whenActionIsDispatched(splitCloseAction({ itemId: ExploreId.right })) .whenActionIsDispatched(splitClose(ExploreId.right))
.thenStateShouldEqual({ .thenStateShouldEqual({
evenSplitPanes: true, evenSplitPanes: true,
panes: { panes: {

View File

@@ -1,9 +1,9 @@
import { createAction } from '@reduxjs/toolkit'; import { createAction } from '@reduxjs/toolkit';
import { AnyAction } from 'redux'; import { AnyAction } from 'redux';
import { ExploreUrlState, serializeStateToUrlParam, SplitOpenOptions, UrlQueryMap } from '@grafana/data'; import { SplitOpenOptions } from '@grafana/data';
import { DataSourceSrv, locationService } from '@grafana/runtime'; import { DataSourceSrv, locationService } from '@grafana/runtime';
import { GetExploreUrlArguments, stopQueryState } from 'app/core/utils/explore'; import { GetExploreUrlArguments } from 'app/core/utils/explore';
import { PanelModel } from 'app/features/dashboard/state'; import { PanelModel } from 'app/features/dashboard/state';
import { ExploreId, ExploreItemState, ExploreState } from 'app/types/explore'; import { ExploreId, ExploreItemState, ExploreState } from 'app/types/explore';
@@ -12,9 +12,10 @@ import { RichHistorySearchFilters, RichHistorySettings } from '../../../core/uti
import { createAsyncThunk, ThunkResult } from '../../../types'; import { createAsyncThunk, ThunkResult } from '../../../types';
import { CorrelationData } from '../../correlations/useCorrelations'; import { CorrelationData } from '../../correlations/useCorrelations';
import { TimeSrv } from '../../dashboard/services/TimeSrv'; import { TimeSrv } from '../../dashboard/services/TimeSrv';
import { withUniqueRefIds } from '../utils/queries';
import { initializeExplore, paneReducer } from './explorePane'; import { initializeExplore, paneReducer } from './explorePane';
import { getUrlStateFromPaneState, makeExplorePaneState } from './utils'; import { DEFAULT_RANGE, makeExplorePaneState } from './utils';
// //
// Actions and Payloads // Actions and Payloads
@@ -50,90 +51,43 @@ export const maximizePaneAction = createAction<{
export const evenPaneResizeAction = createAction('explore/evenPaneResizeAction'); export const evenPaneResizeAction = createAction('explore/evenPaneResizeAction');
/** /**
* Resets state for explore. * Close the pane with the given id.
*/ */
export const resetExploreAction = createAction('explore/resetExplore'); type SplitCloseActionPayload = ExploreId;
export const splitClose = createAction<SplitCloseActionPayload>('explore/splitClose');
/** export interface SetPaneStateActionPayload {
* Close the split view and save URL state. [itemId: string]: Partial<ExploreItemState>;
*/
export interface SplitCloseActionPayload {
itemId: ExploreId;
} }
export const splitCloseAction = createAction<SplitCloseActionPayload>('explore/splitClose'); export const setPaneState = createAction<SetPaneStateActionPayload>('explore/setPaneState');
// export const clearPanes = createAction('explore/clearPanes');
// Action creators
//
/** /**
* Save local redux state back to the URL. Should be called when there is some change that should affect the URL. * Opens a new split pane. It either copies existing state of the left pane
* Not all of the redux state is reflected in URL though. * or uses values from options arg.
*/ *
export const stateSave = (options?: { replace?: boolean }): ThunkResult<void> => { * TODO: this can be improved by better inferring fallback values.
return (_, getState) => {
const { left, right } = getState().explore.panes;
const orgId = getState().user.orgId.toString();
const urlStates: { [index: string]: string | null } = { orgId };
urlStates.left = serializeStateToUrlParam(getUrlStateFromPaneState(left!));
if (right) {
urlStates.right = serializeStateToUrlParam(getUrlStateFromPaneState(right));
} else {
urlStates.right = null;
}
lastSavedUrl.right = urlStates.right;
lastSavedUrl.left = urlStates.left;
locationService.partial({ ...urlStates }, options?.replace);
};
};
// Store the url we saved last se we are not trying to update local state based on that.
export const lastSavedUrl: UrlQueryMap = {};
/**
* Opens a new right split pane by navigating to appropriate URL. It either copies existing state of the left pane
* or uses values from options arg. This does only navigation each pane is then responsible for initialization from
* the URL.
*/ */
export const splitOpen = createAsyncThunk( export const splitOpen = createAsyncThunk(
'explore/splitOpen', 'explore/splitOpen',
async (options: SplitOpenOptions | undefined, { getState }) => { async (options: SplitOpenOptions | undefined, { getState, dispatch }) => {
const leftState: ExploreItemState = getState().explore.panes.left!; const leftState = getState().explore.panes.left;
const leftUrlState = getUrlStateFromPaneState(leftState);
let rightUrlState: ExploreUrlState = leftUrlState;
if (options) { const queries = options?.queries ?? (options?.query ? [options?.query] : leftState?.queries || []);
const { query, queries } = options;
rightUrlState = { await dispatch(
datasource: options.datasourceUid, initializeExplore({
queries: queries ?? (query ? [query] : []), exploreId: ExploreId.right,
range: options.range || leftState.range, datasource: options?.datasourceUid || leftState?.datasourceInstance?.getRef(),
panelsState: options.panelsState, queries: withUniqueRefIds(queries),
}; range: options?.range || leftState?.range.raw || DEFAULT_RANGE,
} panelsState: options?.panelsState || leftState?.panelsState,
})
const urlState = serializeStateToUrlParam(rightUrlState); );
locationService.partial({ right: urlState }, true);
} }
); );
/**
* Close the split view and save URL state. We need to update the state here because when closing we cannot just
* update the URL and let the components handle it because if we swap panes from right to left it is not easily apparent
* from the URL.
*/
export function splitClose(itemId: ExploreId): ThunkResult<void> {
return (dispatch, getState) => {
dispatch(splitCloseAction({ itemId }));
dispatch(stateSave());
};
}
export interface NavigateToExploreDependencies { export interface NavigateToExploreDependencies {
getDataSourceSrv: () => DataSourceSrv; getDataSourceSrv: () => DataSourceSrv;
getTimeSrv: () => TimeSrv; getTimeSrv: () => TimeSrv;
@@ -169,9 +123,7 @@ export const navigateToExplore = (
const initialExploreItemState = makeExplorePaneState(); const initialExploreItemState = makeExplorePaneState();
export const initialExploreState: ExploreState = { export const initialExploreState: ExploreState = {
syncedTimes: false, syncedTimes: false,
panes: { panes: {},
[ExploreId.left]: initialExploreItemState,
},
correlations: undefined, correlations: undefined,
richHistoryStorageFull: false, richHistoryStorageFull: false,
richHistoryLimitExceededWarningShown: false, richHistoryLimitExceededWarningShown: false,
@@ -185,10 +137,9 @@ export const initialExploreState: ExploreState = {
* Actions that have an `exploreId` get routed to the ExploreItemReducer. * Actions that have an `exploreId` get routed to the ExploreItemReducer.
*/ */
export const exploreReducer = (state = initialExploreState, action: AnyAction): ExploreState => { export const exploreReducer = (state = initialExploreState, action: AnyAction): ExploreState => {
if (splitCloseAction.match(action)) { if (splitClose.match(action)) {
const { itemId } = action.payload;
const panes = { const panes = {
left: itemId === ExploreId.left ? state.panes.right : state.panes.left, left: action.payload === ExploreId.left ? state.panes.right : state.panes.left,
}; };
return { return {
...state, ...state,
@@ -254,23 +205,6 @@ export const exploreReducer = (state = initialExploreState, action: AnyAction):
}; };
} }
if (resetExploreAction.match(action)) {
// FIXME: reducers should REALLY not have side effects.
for (const [, pane] of Object.entries(state.panes).filter(([exploreId]) => exploreId !== ExploreId.left)) {
stopQueryState(pane!.querySubscription);
}
return {
...initialExploreState,
panes: {
left: {
...initialExploreItemState,
queries: state.panes.left!.queries,
},
},
};
}
if (richHistorySettingsUpdatedAction.match(action)) { if (richHistorySettingsUpdatedAction.match(action)) {
const richHistorySettings = action.payload; const richHistorySettings = action.payload;
return { return {
@@ -299,14 +233,20 @@ export const exploreReducer = (state = initialExploreState, action: AnyAction):
}; };
} }
if (action.payload) { if (clearPanes.match(action)) {
const { exploreId } = action.payload; return {
if (exploreId !== undefined) { ...state,
panes: {},
};
}
const exploreId: ExploreId | undefined = action.payload?.exploreId;
if (typeof exploreId === 'string') {
return { return {
...state, ...state,
panes: Object.entries(state.panes).reduce<ExploreState['panes']>((acc, [id, pane]) => { panes: Object.entries(state.panes).reduce<ExploreState['panes']>((acc, [id, pane]) => {
if (id === exploreId) { if (id === exploreId) {
acc[id as ExploreId] = paneReducer(pane, action); acc[id] = paneReducer(pane, action);
} else { } else {
acc[id as ExploreId] = pane; acc[id as ExploreId] = pane;
} }
@@ -314,7 +254,6 @@ export const exploreReducer = (state = initialExploreState, action: AnyAction):
}, {}), }, {}),
}; };
} }
}
return state; return state;
}; };

View File

@@ -158,7 +158,7 @@ describe('runQueries', () => {
const { dispatch, getState } = setupTests(); const { dispatch, getState } = setupTests();
setupQueryResponse(getState()); setupQueryResponse(getState());
await dispatch(saveCorrelationsAction([])); await dispatch(saveCorrelationsAction([]));
await dispatch(runQueries(ExploreId.left)); await dispatch(runQueries({ exploreId: ExploreId.left }));
expect(getState().explore.panes.left!.showMetrics).toBeTruthy(); expect(getState().explore.panes.left!.showMetrics).toBeTruthy();
expect(getState().explore.panes.left!.graphResult).toBeDefined(); expect(getState().explore.panes.left!.graphResult).toBeDefined();
}); });
@@ -167,7 +167,7 @@ describe('runQueries', () => {
const { dispatch, getState } = setupTests(); const { dispatch, getState } = setupTests();
setupQueryResponse(getState()); setupQueryResponse(getState());
dispatch(saveCorrelationsAction([])); dispatch(saveCorrelationsAction([]));
dispatch(runQueries(ExploreId.left)); dispatch(runQueries({ exploreId: ExploreId.left }));
const state = getState().explore.panes.left!; const state = getState().explore.panes.left!;
expect(state.queryResponse.request?.requestId).toBe('explore_left'); expect(state.queryResponse.request?.requestId).toBe('explore_left');
@@ -187,7 +187,7 @@ describe('runQueries', () => {
const leftDatasourceInstance = assertIsDefined(getState().explore.panes.left!.datasourceInstance); const leftDatasourceInstance = assertIsDefined(getState().explore.panes.left!.datasourceInstance);
jest.mocked(leftDatasourceInstance.query).mockReturnValueOnce(EMPTY); jest.mocked(leftDatasourceInstance.query).mockReturnValueOnce(EMPTY);
await dispatch(saveCorrelationsAction([])); await dispatch(saveCorrelationsAction([]));
await dispatch(runQueries(ExploreId.left)); await dispatch(runQueries({ exploreId: ExploreId.left }));
await new Promise((resolve) => setTimeout(() => resolve(''), 500)); await new Promise((resolve) => setTimeout(() => resolve(''), 500));
expect(getState().explore.panes.left!.queryResponse.state).toBe(LoadingState.Done); expect(getState().explore.panes.left!.queryResponse.state).toBe(LoadingState.Done);
}); });
@@ -195,7 +195,7 @@ describe('runQueries', () => {
it('shows results only after correlations are loaded', async () => { it('shows results only after correlations are loaded', async () => {
const { dispatch, getState } = setupTests(); const { dispatch, getState } = setupTests();
setupQueryResponse(getState()); setupQueryResponse(getState());
await dispatch(runQueries(ExploreId.left)); await dispatch(runQueries({ exploreId: ExploreId.left }));
expect(getState().explore.panes.left!.graphResult).not.toBeDefined(); expect(getState().explore.panes.left!.graphResult).not.toBeDefined();
await dispatch(saveCorrelationsAction([])); await dispatch(saveCorrelationsAction([]));
expect(getState().explore.panes.left!.graphResult).toBeDefined(); expect(getState().explore.panes.left!.graphResult).toBeDefined();
@@ -791,7 +791,7 @@ describe('reducer', () => {
}); });
it('should cancel any unfinished supplementary queries when a new query is run', async () => { it('should cancel any unfinished supplementary queries when a new query is run', async () => {
dispatch(runQueries(ExploreId.left)); dispatch(runQueries({ exploreId: ExploreId.left }));
// first query is run automatically // first query is run automatically
// loading in progress - subscriptions for both supplementary queries are created, not cleaned up yet // loading in progress - subscriptions for both supplementary queries are created, not cleaned up yet
expect(unsubscribes).toHaveLength(2); expect(unsubscribes).toHaveLength(2);
@@ -799,7 +799,7 @@ describe('reducer', () => {
expect(unsubscribes[1]).not.toBeCalled(); expect(unsubscribes[1]).not.toBeCalled();
setupQueryResponse(getState()); setupQueryResponse(getState());
dispatch(runQueries(ExploreId.left)); dispatch(runQueries({ exploreId: ExploreId.left }));
// a new query is run while supplementary queries are not resolve yet... // a new query is run while supplementary queries are not resolve yet...
expect(unsubscribes[0]).toBeCalled(); expect(unsubscribes[0]).toBeCalled();
expect(unsubscribes[1]).toBeCalled(); expect(unsubscribes[1]).toBeCalled();
@@ -810,7 +810,7 @@ describe('reducer', () => {
}); });
it('should cancel all supported supplementary queries when the main query is canceled', () => { it('should cancel all supported supplementary queries when the main query is canceled', () => {
dispatch(runQueries(ExploreId.left)); dispatch(runQueries({ exploreId: ExploreId.left }));
expect(unsubscribes).toHaveLength(2); expect(unsubscribes).toHaveLength(2);
expect(unsubscribes[0]).not.toBeCalled(); expect(unsubscribes[0]).not.toBeCalled();
expect(unsubscribes[1]).not.toBeCalled(); expect(unsubscribes[1]).not.toBeCalled();
@@ -827,7 +827,7 @@ describe('reducer', () => {
}); });
it('should load supplementary queries after running the query', () => { it('should load supplementary queries after running the query', () => {
dispatch(runQueries(ExploreId.left)); dispatch(runQueries({ exploreId: ExploreId.left }));
expect(unsubscribes).toHaveLength(2); expect(unsubscribes).toHaveLength(2);
}); });
@@ -835,7 +835,7 @@ describe('reducer', () => {
mockDataProvider = () => { mockDataProvider = () => {
return of({ state: LoadingState.Loading, error: undefined, data: [] }); return of({ state: LoadingState.Loading, error: undefined, data: [] });
}; };
dispatch(runQueries(ExploreId.left)); dispatch(runQueries({ exploreId: ExploreId.left }));
for (const type of supplementaryQueryTypes) { for (const type of supplementaryQueryTypes) {
expect(getState().explore.panes.left!.supplementaryQueries[type].data).toBeDefined(); expect(getState().explore.panes.left!.supplementaryQueries[type].data).toBeDefined();
@@ -862,7 +862,7 @@ describe('reducer', () => {
{ state: LoadingState.Done, error: undefined, data: [{}] } { state: LoadingState.Done, error: undefined, data: [{}] }
); );
}; };
dispatch(runQueries(ExploreId.left)); dispatch(runQueries({ exploreId: ExploreId.left }));
for (const types of supplementaryQueryTypes) { for (const types of supplementaryQueryTypes) {
expect(getState().explore.panes.left!.supplementaryQueries[types].data).toBeDefined(); expect(getState().explore.panes.left!.supplementaryQueries[types].data).toBeDefined();
@@ -891,7 +891,7 @@ describe('reducer', () => {
expect(getState().explore.panes.left!.supplementaryQueries[SupplementaryQueryType.LogsSample].enabled).toBe(true); expect(getState().explore.panes.left!.supplementaryQueries[SupplementaryQueryType.LogsSample].enabled).toBe(true);
// verify that if we run a query, it will: 1) not do logs volume, 2) do logs sample 3) provider will still be set for both // verify that if we run a query, it will: 1) not do logs volume, 2) do logs sample 3) provider will still be set for both
dispatch(runQueries(ExploreId.left)); dispatch(runQueries({ exploreId: ExploreId.left }));
expect( expect(
getState().explore.panes.left!.supplementaryQueries[SupplementaryQueryType.LogsVolume].data getState().explore.panes.left!.supplementaryQueries[SupplementaryQueryType.LogsVolume].data
@@ -918,7 +918,7 @@ describe('reducer', () => {
dispatch(setSupplementaryQueryEnabled(ExploreId.left, false, SupplementaryQueryType.LogsSample)); dispatch(setSupplementaryQueryEnabled(ExploreId.left, false, SupplementaryQueryType.LogsSample));
// runQueries sets up providers, but does not run queries // runQueries sets up providers, but does not run queries
dispatch(runQueries(ExploreId.left)); dispatch(runQueries({ exploreId: ExploreId.left }));
expect( expect(
getState().explore.panes.left!.supplementaryQueries[SupplementaryQueryType.LogsVolume].dataProvider getState().explore.panes.left!.supplementaryQueries[SupplementaryQueryType.LogsVolume].dataProvider
).toBeDefined(); ).toBeDefined();

View File

@@ -59,14 +59,9 @@ import {
} from '../utils/supplementaryQueries'; } from '../utils/supplementaryQueries';
import { addHistoryItem, historyUpdatedAction, loadRichHistory } from './history'; import { addHistoryItem, historyUpdatedAction, loadRichHistory } from './history';
import { stateSave } from './main';
import { updateTime } from './time'; import { updateTime } from './time';
import { createCacheKey, getResultsFromCache, filterLogRowsByIndex } from './utils'; import { createCacheKey, getResultsFromCache, filterLogRowsByIndex } from './utils';
//
// Actions and Payloads
//
/** /**
* Adds a query row after the row with the given index. * Adds a query row after the row with the given index.
*/ */
@@ -232,10 +227,6 @@ export interface ClearCachePayload {
} }
export const clearCacheAction = createAction<ClearCachePayload>('explore/clearCache'); export const clearCacheAction = createAction<ClearCachePayload>('explore/clearCache');
//
// Action creators
//
/** /**
* Adds a query row after the row with the given index. * Adds a query row after the row with the given index.
*/ */
@@ -277,7 +268,6 @@ export function cancelQueries(exploreId: ExploreId): ThunkResult<void> {
dispatch(cleanSupplementaryQueryAction({ exploreId, type })); dispatch(cleanSupplementaryQueryAction({ exploreId, type }));
} }
} }
dispatch(stateSave());
}; };
} }
@@ -330,8 +320,8 @@ export const changeQueries = createAsyncThunk<void, ChangeQueriesPayload>(
} }
// if we are removing a query we want to run the remaining ones // if we are removing a query we want to run the remaining ones
if (queries.length < queries.length) { if (queries.length < oldQueries.length) {
dispatch(runQueries(exploreId)); dispatch(runQueries({ exploreId }));
} }
} }
); );
@@ -435,7 +425,7 @@ export function modifyQueries(
dispatch(setQueriesAction({ exploreId, queries: nextQueries })); dispatch(setQueriesAction({ exploreId, queries: nextQueries }));
if (!modification.preventSubmit) { if (!modification.preventSubmit) {
dispatch(runQueries(exploreId)); dispatch(runQueries({ exploreId }));
} }
}; };
} }
@@ -462,21 +452,22 @@ async function handleHistory(
} }
} }
interface RunQueriesOptions {
exploreId: ExploreId;
preserveCache?: boolean;
}
/** /**
* Main action to run queries and dispatches sub-actions based on which result viewers are active * Main action to run queries and dispatches sub-actions based on which result viewers are active
*/ */
export const runQueries = ( export const runQueries = createAsyncThunk<void, RunQueriesOptions>(
exploreId: ExploreId, 'explore/runQueries',
options?: { replaceUrl?: boolean; preserveCache?: boolean } async ({ exploreId, preserveCache }, { dispatch, getState }) => {
): ThunkResult<void> => {
return (dispatch, getState) => {
dispatch(updateTime({ exploreId })); dispatch(updateTime({ exploreId }));
const correlations$ = getCorrelations(); const correlations$ = getCorrelations();
// We always want to clear cache unless we explicitly pass preserveCache parameter // We always want to clear cache unless we explicitly pass preserveCache parameter
const preserveCache = options?.preserveCache === true; if (preserveCache !== true) {
if (!preserveCache) {
dispatch(clearCache(exploreId)); dispatch(clearCache(exploreId));
} }
@@ -506,8 +497,6 @@ export const runQueries = (
handleHistory(dispatch, getState().explore, exploreItemState.history, datasourceInstance, queries, exploreId); handleHistory(dispatch, getState().explore, exploreItemState.history, datasourceInstance, queries, exploreId);
} }
dispatch(stateSave({ replace: options?.replaceUrl }));
const cachedValue = getResultsFromCache(cache, absoluteRange); const cachedValue = getResultsFromCache(cache, absoluteRange);
// If we have results saved in cache, we are going to use those results instead of running queries // If we have results saved in cache, we are going to use those results instead of running queries
@@ -519,21 +508,12 @@ export const runQueries = (
); );
newQuerySubscription = newQuerySource.subscribe((data) => { newQuerySubscription = newQuerySource.subscribe((data) => {
if (!data.error) {
dispatch(stateSave());
}
dispatch(queryStreamUpdatedAction({ exploreId, response: data })); dispatch(queryStreamUpdatedAction({ exploreId, response: data }));
}); });
// If we don't have results saved in cache, run new queries // If we don't have results saved in cache, run new queries
} else { } else {
if (!hasNonEmptyQuery(queries)) { if (!hasNonEmptyQuery(queries) || !datasourceInstance) {
dispatch(stateSave({ replace: options?.replaceUrl })); // Remember to save to state and update location
return;
}
if (!datasourceInstance) {
return; return;
} }
@@ -587,7 +567,7 @@ export const runQueries = (
if (data.state === LoadingState.Done && data.series.length === 0) { if (data.state === LoadingState.Done && data.series.length === 0) {
const range = getShiftedTimeRange(-1, getState().explore.panes[exploreId]!.range); const range = getShiftedTimeRange(-1, getState().explore.panes[exploreId]!.range);
dispatch(updateTime({ exploreId, absoluteRange: range })); dispatch(updateTime({ exploreId, absoluteRange: range }));
dispatch(runQueries(exploreId)); dispatch(runQueries({ exploreId }));
} else { } else {
// We can stop scanning if we have a result // We can stop scanning if we have a result
dispatch(scanStopAction({ exploreId })); dispatch(scanStopAction({ exploreId }));
@@ -635,8 +615,8 @@ export const runQueries = (
} }
dispatch(queryStoreSubscriptionAction({ exploreId, querySubscription: newQuerySubscription })); dispatch(queryStoreSubscriptionAction({ exploreId, querySubscription: newQuerySubscription }));
}; }
}; );
const groupDataQueries = async (datasources: DataQuery[], scopedVars: ScopedVars) => { const groupDataQueries = async (datasources: DataQuery[], scopedVars: ScopedVars) => {
const nonMixedDataSources = datasources.filter((t) => { const nonMixedDataSources = datasources.filter((t) => {
@@ -781,7 +761,7 @@ export function setQueries(exploreId: ExploreId, rawQueries: DataQuery[]): Thunk
const queries = getState().explore.panes[exploreId]!.queries; const queries = getState().explore.panes[exploreId]!.queries;
const nextQueries = rawQueries.map((query, index) => generateNewKeyAndAddRefIdIfMissing(query, queries, index)); const nextQueries = rawQueries.map((query, index) => generateNewKeyAndAddRefIdIfMissing(query, queries, index));
dispatch(setQueriesAction({ exploreId, queries: nextQueries })); dispatch(setQueriesAction({ exploreId, queries: nextQueries }));
dispatch(runQueries(exploreId)); dispatch(runQueries({ exploreId }));
}; };
} }
@@ -798,7 +778,7 @@ export function scanStart(exploreId: ExploreId): ThunkResult<void> {
const range = getShiftedTimeRange(-1, getState().explore.panes[exploreId]!.range); const range = getShiftedTimeRange(-1, getState().explore.panes[exploreId]!.range);
// Set the new range to be displayed // Set the new range to be displayed
dispatch(updateTime({ exploreId, absoluteRange: range })); dispatch(updateTime({ exploreId, absoluteRange: range }));
dispatch(runQueries(exploreId)); dispatch(runQueries({ exploreId }));
}; };
} }

View File

@@ -0,0 +1,19 @@
import { initialExploreState } from './main';
import { selectOrderedExplorePanes } from './selectors';
import { makeExplorePaneState } from './utils';
describe('getOrderedExplorePanes', () => {
it('returns a panes object with entries in the correct order', () => {
const selectorResult = selectOrderedExplorePanes({
explore: {
...initialExploreState,
panes: {
right: makeExplorePaneState(),
left: makeExplorePaneState(),
},
},
});
expect(Object.keys(selectorResult)).toEqual(['left', 'right']);
});
});

View File

@@ -1,5 +1,26 @@
import { ExploreId, StoreState } from 'app/types'; import { createSelector } from '@reduxjs/toolkit';
export const isSplit = (state: StoreState) => Object.keys(state.explore.panes).length > 1; import { ExploreId, ExploreState, StoreState } from 'app/types';
export const getExploreItemSelector = (exploreId: ExploreId) => (state: StoreState) => state.explore.panes[exploreId]; export const selectPanes = (state: Pick<StoreState, 'explore'>) => state.explore.panes;
/**
* Explore renders panes by iterating over the panes object. This selector ensures that entries in the returned panes object
* are in the correct order.
*/
export const selectOrderedExplorePanes = createSelector(selectPanes, (panes) => {
const orderedPanes: ExploreState['panes'] = {};
if (panes.left) {
orderedPanes.left = panes.left;
}
if (panes.right) {
orderedPanes.right = panes.right;
}
return orderedPanes;
});
export const isSplit = createSelector(selectPanes, (panes) => Object.keys(panes).length > 1);
export const getExploreItemSelector = (exploreId: ExploreId) =>
createSelector(selectPanes, (panes) => panes[exploreId]);

View File

@@ -4,10 +4,8 @@ import { dateTime, LoadingState } from '@grafana/data';
import { configureStore } from 'app/store/configureStore'; import { configureStore } from 'app/store/configureStore';
import { ExploreId, ExploreItemState } from 'app/types'; import { ExploreId, ExploreItemState } from 'app/types';
import { silenceConsoleOutput } from '../../../../test/core/utils/silenceConsoleOutput';
import { createDefaultInitialState } from './helpers'; import { createDefaultInitialState } from './helpers';
import { changeRangeAction, changeRefreshIntervalAction, timeReducer, updateTime } from './time'; import { changeRangeAction, changeRefreshInterval, timeReducer, updateTime } from './time';
import { makeExplorePaneState } from './utils'; import { makeExplorePaneState } from './utils';
const MOCK_TIME_RANGE = {}; const MOCK_TIME_RANGE = {};
@@ -30,14 +28,10 @@ jest.mock('@grafana/runtime', () => ({
})); }));
describe('Explore item reducer', () => { describe('Explore item reducer', () => {
silenceConsoleOutput();
describe('When time is updated', () => { describe('When time is updated', () => {
it('Time service is re-initialized and template service is updated with the new time range', async () => { it('Time service is re-initialized and template service is updated with the new time range', async () => {
const { dispatch } = configureStore({ const { dispatch } = configureStore(createDefaultInitialState().defaultInitialState as any);
...(createDefaultInitialState() as any), dispatch(updateTime({ exploreId: ExploreId.left }));
});
await dispatch(updateTime({ exploreId: ExploreId.left }));
expect(mockTimeSrv.init).toBeCalled(); expect(mockTimeSrv.init).toBeCalled();
expect(mockTemplateSrv.updateTimeRange).toBeCalledWith(MOCK_TIME_RANGE); expect(mockTemplateSrv.updateTimeRange).toBeCalledWith(MOCK_TIME_RANGE);
}); });
@@ -62,7 +56,7 @@ describe('Explore item reducer', () => {
}; };
reducerTester<ExploreItemState>() reducerTester<ExploreItemState>()
.givenReducer(timeReducer, initialState) .givenReducer(timeReducer, initialState)
.whenActionIsDispatched(changeRefreshIntervalAction({ exploreId: ExploreId.left, refreshInterval: 'LIVE' })) .whenActionIsDispatched(changeRefreshInterval({ exploreId: ExploreId.left, refreshInterval: 'LIVE' }))
.thenStateShouldEqual(expectedState); .thenStateShouldEqual(expectedState);
}); });
@@ -82,7 +76,7 @@ describe('Explore item reducer', () => {
}; };
reducerTester<ExploreItemState>() reducerTester<ExploreItemState>()
.givenReducer(timeReducer, initialState) .givenReducer(timeReducer, initialState)
.whenActionIsDispatched(changeRefreshIntervalAction({ exploreId: ExploreId.left, refreshInterval: '' })) .whenActionIsDispatched(changeRefreshInterval({ exploreId: ExploreId.left, refreshInterval: '' }))
.thenStateShouldEqual(expectedState); .thenStateShouldEqual(expectedState);
}); });
}); });

View File

@@ -1,4 +1,4 @@
import { AnyAction, createAction, PayloadAction } from '@reduxjs/toolkit'; import { AnyAction, createAction } from '@reduxjs/toolkit';
import { AbsoluteTimeRange, dateTimeForTimeZone, LoadingState, RawTimeRange, TimeRange } from '@grafana/data'; import { AbsoluteTimeRange, dateTimeForTimeZone, LoadingState, RawTimeRange, TimeRange } from '@grafana/data';
import { getTemplateSrv } from '@grafana/runtime'; import { getTemplateSrv } from '@grafana/runtime';
@@ -12,7 +12,7 @@ import { ExploreId } from 'app/types/explore';
import { getTimeSrv } from '../../dashboard/services/TimeSrv'; import { getTimeSrv } from '../../dashboard/services/TimeSrv';
import { TimeModel } from '../../dashboard/state/TimeModel'; import { TimeModel } from '../../dashboard/state/TimeModel';
import { syncTimesAction, stateSave } from './main'; import { syncTimesAction } from './main';
import { runQueries } from './query'; import { runQueries } from './query';
// //
@@ -33,7 +33,7 @@ export interface ChangeRefreshIntervalPayload {
exploreId: ExploreId; exploreId: ExploreId;
refreshInterval: string; refreshInterval: string;
} }
export const changeRefreshIntervalAction = createAction<ChangeRefreshIntervalPayload>('explore/changeRefreshInterval'); export const changeRefreshInterval = createAction<ChangeRefreshIntervalPayload>('explore/changeRefreshInterval');
export const updateTimeRange = (options: { export const updateTimeRange = (options: {
exploreId: ExploreId; exploreId: ExploreId;
@@ -45,25 +45,15 @@ export const updateTimeRange = (options: {
if (syncedTimes) { if (syncedTimes) {
Object.keys(getState().explore.panes).forEach((exploreId) => { Object.keys(getState().explore.panes).forEach((exploreId) => {
dispatch(updateTime({ ...options, exploreId: exploreId as ExploreId })); dispatch(updateTime({ ...options, exploreId: exploreId as ExploreId }));
dispatch(runQueries(exploreId as ExploreId, { preserveCache: true })); dispatch(runQueries({ exploreId: exploreId as ExploreId, preserveCache: true }));
}); });
} else { } else {
dispatch(updateTime({ ...options })); dispatch(updateTime({ ...options }));
dispatch(runQueries(options.exploreId, { preserveCache: true })); dispatch(runQueries({ exploreId: options.exploreId, preserveCache: true }));
} }
}; };
}; };
/**
* Change the refresh interval of Explore. Called from the Refresh picker.
*/
export function changeRefreshInterval(
exploreId: ExploreId,
refreshInterval: string
): PayloadAction<ChangeRefreshIntervalPayload> {
return changeRefreshIntervalAction({ exploreId, refreshInterval });
}
export const updateTime = (config: { export const updateTime = (config: {
exploreId: ExploreId; exploreId: ExploreId;
rawRange?: RawTimeRange; rawRange?: RawTimeRange;
@@ -126,7 +116,6 @@ export function syncTimes(exploreId: ExploreId): ThunkResult<void> {
const isTimeSynced = getState().explore.syncedTimes; const isTimeSynced = getState().explore.syncedTimes;
dispatch(syncTimesAction({ syncedTimes: !isTimeSynced })); dispatch(syncTimesAction({ syncedTimes: !isTimeSynced }));
dispatch(stateSave());
}; };
} }
@@ -145,8 +134,6 @@ export function makeAbsoluteTime(): ThunkResult<void> {
const absoluteRange: AbsoluteTimeRange = { from: range.from.valueOf(), to: range.to.valueOf() }; const absoluteRange: AbsoluteTimeRange = { from: range.from.valueOf(), to: range.to.valueOf() };
dispatch(updateTime({ exploreId: exploreId as ExploreId, absoluteRange })); dispatch(updateTime({ exploreId: exploreId as ExploreId, absoluteRange }));
}); });
dispatch(stateSave());
}; };
} }
@@ -159,7 +146,7 @@ export function makeAbsoluteTime(): ThunkResult<void> {
// the frozen state. // the frozen state.
// https://github.com/reduxjs/redux-toolkit/issues/242 // https://github.com/reduxjs/redux-toolkit/issues/242
export const timeReducer = (state: ExploreItemState, action: AnyAction): ExploreItemState => { export const timeReducer = (state: ExploreItemState, action: AnyAction): ExploreItemState => {
if (changeRefreshIntervalAction.match(action)) { if (changeRefreshInterval.match(action)) {
const { refreshInterval } = action.payload; const { refreshInterval } = action.payload;
const live = RefreshPicker.isLive(refreshInterval); const live = RefreshPicker.isLive(refreshInterval);
const sortOrder = refreshIntervalToSortOrder(refreshInterval); const sortOrder = refreshIntervalToSortOrder(refreshInterval);

View File

@@ -1,5 +1,5 @@
import store from '../../../core/store'; import { dateTime } from '@grafana/data';
import { lastUsedDatasourceKeyForOrgId } from '../../../core/utils/explore'; import * as exploreUtils from 'app/core/utils/explore';
const dataSourceMock = { const dataSourceMock = {
get: jest.fn(), get: jest.fn(),
@@ -8,19 +8,24 @@ jest.mock('app/features/plugins/datasource_srv', () => ({
getDatasourceSrv: jest.fn(() => dataSourceMock), getDatasourceSrv: jest.fn(() => dataSourceMock),
})); }));
jest.spyOn(store, 'set'); import { loadAndInitDatasource, getRange } from './utils';
import { loadAndInitDatasource } from './utils';
const DEFAULT_DATASOURCE = { uid: 'abc123', name: 'Default' }; const DEFAULT_DATASOURCE = { uid: 'abc123', name: 'Default' };
const TEST_DATASOURCE = { uid: 'def789', name: 'Test' }; const TEST_DATASOURCE = { uid: 'def789', name: 'Test' };
describe('loadAndInitDatasource', () => { describe('loadAndInitDatasource', () => {
beforeEach(() => { let setLastUsedDatasourceUIDSpy;
afterEach(() => {
jest.clearAllMocks(); jest.clearAllMocks();
}); });
afterAll(() => {
jest.restoreAllMocks();
});
it('falls back to default datasource if the provided one was not found', async () => { it('falls back to default datasource if the provided one was not found', async () => {
setLastUsedDatasourceUIDSpy = jest.spyOn(exploreUtils, 'setLastUsedDatasourceUID');
dataSourceMock.get.mockRejectedValueOnce(new Error('Datasource not found')); dataSourceMock.get.mockRejectedValueOnce(new Error('Datasource not found'));
dataSourceMock.get.mockResolvedValue(DEFAULT_DATASOURCE); dataSourceMock.get.mockResolvedValue(DEFAULT_DATASOURCE);
@@ -30,10 +35,11 @@ describe('loadAndInitDatasource', () => {
expect(dataSourceMock.get).toBeCalledWith({ uid: 'Unknown' }); expect(dataSourceMock.get).toBeCalledWith({ uid: 'Unknown' });
expect(dataSourceMock.get).toBeCalledWith(); expect(dataSourceMock.get).toBeCalledWith();
expect(instance).toMatchObject(DEFAULT_DATASOURCE); expect(instance).toMatchObject(DEFAULT_DATASOURCE);
expect(store.set).toBeCalledWith(lastUsedDatasourceKeyForOrgId(1), DEFAULT_DATASOURCE.uid); expect(setLastUsedDatasourceUIDSpy).toBeCalledWith(1, DEFAULT_DATASOURCE.uid);
}); });
it('saves last loaded data source uid', async () => { it('saves last loaded data source uid', async () => {
setLastUsedDatasourceUIDSpy = jest.spyOn(exploreUtils, 'setLastUsedDatasourceUID');
dataSourceMock.get.mockResolvedValue(TEST_DATASOURCE); dataSourceMock.get.mockResolvedValue(TEST_DATASOURCE);
const { instance } = await loadAndInitDatasource(1, { uid: 'Test' }); const { instance } = await loadAndInitDatasource(1, { uid: 'Test' });
@@ -41,6 +47,39 @@ describe('loadAndInitDatasource', () => {
expect(dataSourceMock.get).toBeCalledTimes(1); expect(dataSourceMock.get).toBeCalledTimes(1);
expect(dataSourceMock.get).toBeCalledWith({ uid: 'Test' }); expect(dataSourceMock.get).toBeCalledWith({ uid: 'Test' });
expect(instance).toMatchObject(TEST_DATASOURCE); expect(instance).toMatchObject(TEST_DATASOURCE);
expect(store.set).toBeCalledWith(lastUsedDatasourceKeyForOrgId(1), TEST_DATASOURCE.uid); expect(setLastUsedDatasourceUIDSpy).toBeCalledWith(1, TEST_DATASOURCE.uid);
});
});
describe('getRange', () => {
it('should parse moment date', () => {
// convert date strings to moment object
const range = { from: dateTime('2020-10-22T10:44:33.615Z'), to: dateTime('2020-10-22T10:49:33.615Z') };
const result = getRange(range, 'browser');
expect(result.raw).toEqual(range);
});
it('should parse epoch strings', () => {
const range = {
from: dateTime('2020-10-22T10:00:00Z').valueOf().toString(),
to: dateTime('2020-10-22T11:00:00Z').valueOf().toString(),
};
const result = getRange(range, 'browser');
expect(result.from.valueOf()).toEqual(dateTime('2020-10-22T10:00:00Z').valueOf());
expect(result.to.valueOf()).toEqual(dateTime('2020-10-22T11:00:00Z').valueOf());
expect(result.raw.from.valueOf()).toEqual(dateTime('2020-10-22T10:00:00Z').valueOf());
expect(result.raw.to.valueOf()).toEqual(dateTime('2020-10-22T11:00:00Z').valueOf());
});
it('should parse ISO strings', () => {
const range = {
from: dateTime('2020-10-22T10:00:00Z').toISOString(),
to: dateTime('2020-10-22T11:00:00Z').toISOString(),
};
const result = getRange(range, 'browser');
expect(result.from.valueOf()).toEqual(dateTime('2020-10-22T10:00:00Z').valueOf());
expect(result.to.valueOf()).toEqual(dateTime('2020-10-22T11:00:00Z').valueOf());
expect(result.raw.from.valueOf()).toEqual(dateTime('2020-10-22T10:00:00Z').valueOf());
expect(result.raw.to.valueOf()).toEqual(dateTime('2020-10-22T11:00:00Z').valueOf());
}); });
}); });

View File

@@ -1,25 +1,28 @@
import { isEmpty, isObject, mapValues, omitBy } from 'lodash';
import { import {
AbsoluteTimeRange, AbsoluteTimeRange,
DataSourceApi, DataSourceApi,
EventBusExtended, EventBusExtended,
ExploreUrlState,
getDefaultTimeRange, getDefaultTimeRange,
HistoryItem, HistoryItem,
LoadingState, LoadingState,
LogRowModel, LogRowModel,
PanelData, PanelData,
RawTimeRange,
TimeFragment,
TimeRange,
dateMath,
DateTime,
isDateTime,
toUtc,
} from '@grafana/data'; } from '@grafana/data';
import { DataSourceRef } from '@grafana/schema'; import { DataSourceRef, TimeZone } from '@grafana/schema';
import { ExplorePanelData } from 'app/types'; import { ExplorePanelData } from 'app/types';
import { ExploreItemState } from 'app/types/explore'; import { ExploreItemState } from 'app/types/explore';
import store from '../../../core/store'; import store from '../../../core/store';
import { clearQueryKeys, lastUsedDatasourceKeyForOrgId } from '../../../core/utils/explore'; import { setLastUsedDatasourceUID } from '../../../core/utils/explore';
import { getDatasourceSrv } from '../../plugins/datasource_srv'; import { getDatasourceSrv } from '../../plugins/datasource_srv';
import { loadSupplementaryQueries } from '../utils/supplementaryQueries'; import { loadSupplementaryQueries } from '../utils/supplementaryQueries';
import { toRawTimeRange } from '../utils/time';
export const DEFAULT_RANGE = { export const DEFAULT_RANGE = {
from: 'now-6h', from: 'now-6h',
@@ -37,7 +40,6 @@ export const storeGraphStyle = (graphStyle: string): void => {
export const makeExplorePaneState = (): ExploreItemState => ({ export const makeExplorePaneState = (): ExploreItemState => ({
containerWidth: 0, containerWidth: 0,
datasourceInstance: null, datasourceInstance: null,
datasourceMissing: false,
history: [], history: [],
queries: [], queries: [],
initialized: false, initialized: false,
@@ -112,33 +114,10 @@ export async function loadAndInitDatasource(
const history = store.getObject<HistoryItem[]>(historyKey, []); const history = store.getObject<HistoryItem[]>(historyKey, []);
// Save last-used datasource // Save last-used datasource
store.set(lastUsedDatasourceKeyForOrgId(orgId), instance.uid); setLastUsedDatasourceUID(orgId, instance.uid);
return { history, instance }; return { history, instance };
} }
// recursively walks an object, removing keys where the value is undefined
// if the resulting object is empty, returns undefined
function pruneObject(obj: object): object | undefined {
let pruned = mapValues(obj, (value) => (isObject(value) ? pruneObject(value) : value));
pruned = omitBy<typeof pruned>(pruned, isEmpty);
if (isEmpty(pruned)) {
return undefined;
}
return pruned;
}
export function getUrlStateFromPaneState(pane: ExploreItemState): ExploreUrlState {
return {
// datasourceInstance should not be undefined anymore here but in case there is some path for it to be undefined
// lets just fallback instead of crashing.
datasource: pane.datasourceInstance?.uid || '',
queries: pane.queries.map(clearQueryKeys),
range: toRawTimeRange(pane.range),
// don't include panelsState in the url unless a piece of state is actually set
panelsState: pruneObject(pane.panelsState),
};
}
export function createCacheKey(absRange: AbsoluteTimeRange) { export function createCacheKey(absRange: AbsoluteTimeRange) {
const params = { const params = {
from: absRange.from, from: absRange.from,
@@ -161,6 +140,57 @@ export function getResultsFromCache(
return cacheValue; return cacheValue;
} }
export function getRange(range: RawTimeRange, timeZone: TimeZone): TimeRange {
const raw = {
from: parseRawTime(range.from)!,
to: parseRawTime(range.to)!,
};
return {
from: dateMath.parse(raw.from, false, timeZone)!,
to: dateMath.parse(raw.to, true, timeZone)!,
raw,
};
}
function parseRawTime(value: string | DateTime): TimeFragment | null {
if (value === null) {
return null;
}
if (isDateTime(value)) {
return value;
}
if (value.indexOf('now') !== -1) {
return value;
}
if (value.length === 8) {
return toUtc(value, 'YYYYMMDD');
}
if (value.length === 15) {
return toUtc(value, 'YYYYMMDDTHHmmss');
}
// Backward compatibility
if (value.length === 19) {
return toUtc(value, 'YYYY-MM-DD HH:mm:ss');
}
// This should handle cases where value is an epoch time as string
if (value.match(/^\d+$/)) {
const epoch = parseInt(value, 10);
return toUtc(epoch);
}
// This should handle ISO strings
const time = toUtc(value);
if (time.isValid()) {
return time;
}
return null;
}
export const filterLogRowsByIndex = ( export const filterLogRowsByIndex = (
clearedAtIndex: ExploreItemState['clearedAtIndex'], clearedAtIndex: ExploreItemState['clearedAtIndex'],
logRows?: LogRowModel[] logRows?: LogRowModel[]

View File

@@ -30,12 +30,12 @@ export function useLiveTailControls(exploreId: ExploreId) {
// TODO referencing this from perspective of refresh picker when there is designated button for it now is not // TODO referencing this from perspective of refresh picker when there is designated button for it now is not
// great. Needs a bit of refactoring. // great. Needs a bit of refactoring.
dispatch(changeRefreshInterval(exploreId, RefreshPicker.offOption.value)); dispatch(changeRefreshInterval({ exploreId, refreshInterval: RefreshPicker.offOption.value }));
dispatch(runQueries(exploreId)); dispatch(runQueries({ exploreId }));
}, [exploreId, dispatch, pause]); }, [exploreId, dispatch, pause]);
const start = useCallback(() => { const start = useCallback(() => {
dispatch(changeRefreshInterval(exploreId, RefreshPicker.liveOption.value)); dispatch(changeRefreshInterval({ exploreId, refreshInterval: RefreshPicker.liveOption.value }));
}, [exploreId, dispatch]); }, [exploreId, dispatch]);
const clear = useCallback(() => { const clear = useCallback(() => {

View File

@@ -0,0 +1,32 @@
import { DataQuery } from '@grafana/schema';
import { getNextRefIdChar } from 'app/core/utils/query';
/**
* Makes sure all the queries have unique (and valid) refIds
*/
export function withUniqueRefIds(queries: DataQuery[]): DataQuery[] {
const refIds = new Set<string>(queries.map((query) => query.refId).filter(Boolean));
if (refIds.size === queries.length) {
return queries;
}
refIds.clear();
return queries.map((query) => {
if (query.refId && !refIds.has(query.refId)) {
refIds.add(query.refId);
return query;
}
const refId = getNextRefIdChar(queries);
refIds.add(refId);
const newQuery = {
...query,
refId,
};
return newQuery;
});
}

View File

@@ -1,18 +0,0 @@
import { isDateTime, RawTimeRange, TimeRange } from '@grafana/data';
export const toRawTimeRange = (range: TimeRange): RawTimeRange => {
let from = range.raw.from;
if (isDateTime(from)) {
from = from.valueOf().toString(10);
}
let to = range.raw.to;
if (isDateTime(to)) {
to = to.valueOf().toString(10);
}
return {
from,
to,
};
};

View File

@@ -1,4 +1,4 @@
import { configureStore as reduxConfigureStore } from '@reduxjs/toolkit'; import { configureStore as reduxConfigureStore, createListenerMiddleware } from '@reduxjs/toolkit';
import { browseDashboardsAPI } from 'app/features/browse-dashboards/api/browseDashboardsAPI'; import { browseDashboardsAPI } from 'app/features/browse-dashboards/api/browseDashboardsAPI';
import { publicDashboardApi } from 'app/features/dashboard/api/publicDashboardApi'; import { publicDashboardApi } from 'app/features/dashboard/api/publicDashboardApi';
@@ -17,11 +17,14 @@ export function addRootReducer(reducers: any) {
addReducer(reducers); addReducer(reducers);
} }
const listenerMiddleware = createListenerMiddleware();
export function configureStore(initialState?: Partial<StoreState>) { export function configureStore(initialState?: Partial<StoreState>) {
const store = reduxConfigureStore({ const store = reduxConfigureStore({
reducer: createRootReducer(), reducer: createRootReducer(),
middleware: (getDefaultMiddleware) => middleware: (getDefaultMiddleware) =>
getDefaultMiddleware({ thunk: true, serializableCheck: false, immutableCheck: false }).concat( getDefaultMiddleware({ thunk: true, serializableCheck: false, immutableCheck: false }).concat(
listenerMiddleware.middleware,
alertingApi.middleware, alertingApi.middleware,
publicDashboardApi.middleware, publicDashboardApi.middleware,
browseDashboardsAPI.middleware browseDashboardsAPI.middleware

View File

@@ -26,8 +26,8 @@ export enum ExploreId {
} }
export type ExploreQueryParams = { export type ExploreQueryParams = {
left: string; left?: string;
right: string; right?: string;
}; };
/** /**
@@ -93,10 +93,6 @@ export interface ExploreItemState {
* Datasource instance that has been selected. Datasource-specific logic can be run on this object. * Datasource instance that has been selected. Datasource-specific logic can be run on this object.
*/ */
datasourceInstance?: DataSourceApi | null; datasourceInstance?: DataSourceApi | null;
/**
* True if there is no datasource to be selected.
*/
datasourceMissing: boolean;
/** /**
* Emitter to send events to the rest of Grafana. * Emitter to send events to the rest of Grafana.
*/ */
@@ -211,8 +207,6 @@ export interface ExploreItemState {
supplementaryQueries: SupplementaryQueries; supplementaryQueries: SupplementaryQueries;
panelsState: ExplorePanelsState; panelsState: ExplorePanelsState;
isFromCompactUrl?: boolean;
} }
export interface ExploreUpdateState { export interface ExploreUpdateState {

View File

@@ -1,11 +1,13 @@
/* eslint-disable no-restricted-imports */ /* eslint-disable no-restricted-imports */
import { import {
Action, Action,
addListener as addListenerUntyped,
AsyncThunk, AsyncThunk,
AsyncThunkOptions, AsyncThunkOptions,
AsyncThunkPayloadCreator, AsyncThunkPayloadCreator,
createAsyncThunk as createAsyncThunkUntyped, createAsyncThunk as createAsyncThunkUntyped,
PayloadAction, PayloadAction,
TypedAddListener,
} from '@reduxjs/toolkit'; } from '@reduxjs/toolkit';
import { import {
useSelector as useSelectorUntyped, useSelector as useSelectorUntyped,
@@ -37,3 +39,5 @@ export const createAsyncThunk = <Returned, ThunkArg = void, ThunkApiConfig exten
options?: AsyncThunkOptions<ThunkArg, ThunkApiConfig> options?: AsyncThunkOptions<ThunkArg, ThunkApiConfig>
): AsyncThunk<Returned, ThunkArg, ThunkApiConfig> => ): AsyncThunk<Returned, ThunkArg, ThunkApiConfig> =>
createAsyncThunkUntyped<Returned, ThunkArg, ThunkApiConfig>(typePrefix, payloadCreator, options); createAsyncThunkUntyped<Returned, ThunkArg, ThunkApiConfig>(typePrefix, payloadCreator, options);
export const addListener = addListenerUntyped as TypedAddListener<RootState, AppDispatch>;