mirror of
https://github.com/grafana/grafana.git
synced 2025-02-16 02:23:31 -06:00
* Basic implementation of getLogsVolumeQuery method * Add todos * Add a switcher to automatically load logs volume * De-scope dismissing logs volume panel * De-scope logs volume query cancellation * Remove todo * Aggregate logs volume components in single panel * Show logs volume only when it's available * Aggregate logs volume by level * Simplify aggregation * Handle no logs volume data * Add error handling * Do not show auto-load logs volume switcher when loading logs volume is not available * Remove old logs volume graph * Clean up * Make getting data provider more generic * Provide complete logs volume data (error, isLoading) * Display more specific error message * Add missing props to mocks * Remove setRequest method * Mark getQueryRelatedDataProviders as internal * Add missing dataQueryRequest and add a todo * Remove redundant loading state * Do not mutate existing queries * Apply fix for zooming-in from main * Post-merge fixes * Create collection for data provider results * Use more generic names * Move aggregation logic to Loki logs volume provider * Move LogsVolume to common types * Update tests * Post-merge fixes * Fix mapping related data values * Simplify prop mappings * Add docs * Fix property name * Clean-up * Mark new types as internal * Reduce number of providers to logs volume only * Simplify data structure to DataQueryResponse * Move Logs Volume panel to a separate component * Test logsVolumeProvider.ts * Add observable version of datasource mock * Test getLogsVolumeDataProvider method * Test LogsVolumePanel * Test logs volume reducer * Clean up * Clean up * Fix test * Use sum by to use level field directly * Fix strict type errors * Fix strict type errors * Use "logs" instead of "unknown" if only one level was detected * Add docs about logs volume * Rename histogramRequest to logsVolumeRequest * Use LogsVolumeContentWrapper all content types * Move `autoLoadLogsVolume` local storage handling * Fix strict error * Move getting autoLoadLogsVolume to initial state * Cancel current logs volume subscription * Test cancelling subscriptions * Update docs/sources/datasources/loki.md Co-authored-by: achatterjee-grafana <70489351+achatterjee-grafana@users.noreply.github.com> * Update packages/grafana-data/src/types/explore.ts Co-authored-by: achatterjee-grafana <70489351+achatterjee-grafana@users.noreply.github.com> * Inline container styles * Ensure logs volume is aggregated per each subscription separately * Simplify logs volume provider * Type-guard support for logs volume provider * Simplify event handlers to avoid casting * Clean up and docs * Move auto-load switcher to logs volume panel * Fix test * Move DataSourceWithLogsVolumeSupport to avoid cross referencing * Simplify interface * Bring back old histogram and hide the new one behind a feature flag * Add missing props to logs histogram panel * Clean up the provider when it's not supported * Simplify storing autoLoadLogsVolume * Remove docs * Update packages/grafana-data/src/types/logsVolume.ts Co-authored-by: Andrej Ocenas <mr.ocenas@gmail.com> * Skip dataframes without fields (instant queries) * Revert styles changes * Revert styles changes * Add release tag Co-authored-by: achatterjee-grafana <70489351+achatterjee-grafana@users.noreply.github.com> Co-authored-by: Andrej Ocenas <mr.ocenas@gmail.com>
124 lines
3.3 KiB
TypeScript
124 lines
3.3 KiB
TypeScript
import {
|
|
AbsoluteTimeRange,
|
|
DataSourceApi,
|
|
EventBusExtended,
|
|
ExploreUrlState,
|
|
getDefaultTimeRange,
|
|
HistoryItem,
|
|
LoadingState,
|
|
PanelData,
|
|
} from '@grafana/data';
|
|
|
|
import { ExploreItemState } from 'app/types/explore';
|
|
import { getDatasourceSrv } from '../../plugins/datasource_srv';
|
|
import store from '../../../core/store';
|
|
import { clearQueryKeys, lastUsedDatasourceKeyForOrgId } from '../../../core/utils/explore';
|
|
import { toRawTimeRange } from '../utils/time';
|
|
|
|
export const DEFAULT_RANGE = {
|
|
from: 'now-6h',
|
|
to: 'now',
|
|
};
|
|
|
|
/**
|
|
* Returns a fresh Explore area state
|
|
*/
|
|
export const makeExplorePaneState = (): ExploreItemState => ({
|
|
containerWidth: 0,
|
|
datasourceInstance: null,
|
|
datasourceMissing: false,
|
|
history: [],
|
|
queries: [],
|
|
initialized: false,
|
|
range: {
|
|
from: null,
|
|
to: null,
|
|
raw: DEFAULT_RANGE,
|
|
} as any,
|
|
absoluteRange: {
|
|
from: null,
|
|
to: null,
|
|
} as any,
|
|
scanning: false,
|
|
loading: false,
|
|
queryKeys: [],
|
|
isLive: false,
|
|
isPaused: false,
|
|
queryResponse: createEmptyQueryResponse(),
|
|
tableResult: null,
|
|
graphResult: null,
|
|
logsResult: null,
|
|
eventBridge: (null as unknown) as EventBusExtended,
|
|
cache: [],
|
|
logsVolumeDataProvider: undefined,
|
|
logsVolumeData: undefined,
|
|
});
|
|
|
|
export const createEmptyQueryResponse = (): PanelData => ({
|
|
state: LoadingState.NotStarted,
|
|
series: [],
|
|
timeRange: getDefaultTimeRange(),
|
|
});
|
|
|
|
export async function loadAndInitDatasource(
|
|
orgId: number,
|
|
datasourceUid?: string
|
|
): Promise<{ history: HistoryItem[]; instance: DataSourceApi }> {
|
|
let instance;
|
|
try {
|
|
instance = await getDatasourceSrv().get(datasourceUid);
|
|
} catch (error) {
|
|
// Falling back to the default data source in case the provided data source was not found.
|
|
// It may happen if last used data source or the data source provided in the URL has been
|
|
// removed or it is not provisioned anymore.
|
|
instance = await getDatasourceSrv().get();
|
|
}
|
|
if (instance.init) {
|
|
try {
|
|
instance.init();
|
|
} catch (err) {
|
|
// TODO: should probably be handled better
|
|
console.error(err);
|
|
}
|
|
}
|
|
|
|
const historyKey = `grafana.explore.history.${instance.meta?.id}`;
|
|
const history = store.getObject(historyKey, []);
|
|
// Save last-used datasource
|
|
|
|
store.set(lastUsedDatasourceKeyForOrgId(orgId), instance.uid);
|
|
return { history, instance };
|
|
}
|
|
|
|
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?.name || '',
|
|
queries: pane.queries.map(clearQueryKeys),
|
|
range: toRawTimeRange(pane.range),
|
|
};
|
|
}
|
|
|
|
export function createCacheKey(absRange: AbsoluteTimeRange) {
|
|
const params = {
|
|
from: absRange.from,
|
|
to: absRange.to,
|
|
};
|
|
|
|
const cacheKey = Object.entries(params)
|
|
.map(([k, v]) => `${encodeURIComponent(k)}=${encodeURIComponent(v.toString())}`)
|
|
.join('&');
|
|
return cacheKey;
|
|
}
|
|
|
|
export function getResultsFromCache(
|
|
cache: Array<{ key: string; value: PanelData }>,
|
|
absoluteRange: AbsoluteTimeRange
|
|
): PanelData | undefined {
|
|
const cacheKey = createCacheKey(absoluteRange);
|
|
const cacheIdx = cache.findIndex((c) => c.key === cacheKey);
|
|
const cacheValue = cacheIdx >= 0 ? cache[cacheIdx].value : undefined;
|
|
return cacheValue;
|
|
}
|