import type {PageFilters} from 'sentry/types'; import EventView from 'sentry/utils/discover/eventView'; import type {Sort} from 'sentry/utils/discover/fields'; import {DiscoverDatasets} from 'sentry/utils/discover/types'; import {MutableSearch} from 'sentry/utils/tokenizeSearch'; import usePageFilters from 'sentry/utils/usePageFilters'; import type { MetricsProperty, MetricsResponse, SpanMetricsQueryFilters, } from 'sentry/views/starfish/types'; import {useWrappedDiscoverQuery} from 'sentry/views/starfish/utils/useSpansQuery'; interface UseSpanMetricsOptions { cursor?: string; enabled?: boolean; fields?: Fields; filters?: SpanMetricsQueryFilters; limit?: number; referrer?: string; sorts?: Sort[]; } export const useSpanMetrics = ( options: UseSpanMetricsOptions = {} ) => { const {fields = [], filters = {}, sorts = [], limit, cursor, referrer} = options; const pageFilters = usePageFilters(); const eventView = getEventView(filters, fields, sorts, pageFilters.selection); // TODO: Checking that every filter has a value might not be a good choice, since this API is not convenient. Instead, it's probably fine to omit keys with blank values const enabled = options.enabled && Object.values(filters).every(value => Boolean(value)); const result = useWrappedDiscoverQuery({ eventView, initialData: [], limit, enabled, referrer, cursor, }); // This type is a little awkward but it explicitly states that the response could be empty. This doesn't enable unchecked access errors, but it at least indicates that it's possible that there's no data // eslint-disable-next-line @typescript-eslint/ban-types const data = (result?.data ?? []) as Pick[] | []; return { ...result, data, isEnabled: enabled, }; }; function getEventView( filters: SpanMetricsQueryFilters = {}, fields: string[] = [], sorts: Sort[] = [], pageFilters: PageFilters ) { const query = MutableSearch.fromQueryObject(filters); const eventView = EventView.fromNewQueryWithPageFilters( { name: '', query: query.formatString(), fields, dataset: DiscoverDatasets.SPANS_METRICS, version: 2, }, pageFilters ); if (sorts.length > 0) { eventView.sorts = sorts; } return eventView; }