import {useCallback, useEffect, useMemo} from 'react'; import { getDiffInMinutes, GranularityLadder, ONE_HOUR, ONE_WEEK, SIXTY_DAYS, THIRTY_DAYS, TWENTY_FOUR_HOURS, TWO_WEEKS, } from 'sentry/components/charts/utils'; import {parseStatsPeriod} from 'sentry/components/organizations/pageFilters/parse'; import {t} from 'sentry/locale'; import type {PageFilters} from 'sentry/types/core'; import {parsePeriodToHours} from 'sentry/utils/duration/parsePeriodToHours'; import {useUpdateQuery} from 'sentry/utils/metrics'; import {decodeScalar} from 'sentry/utils/queryString'; import useLocationQuery from 'sentry/utils/url/useLocationQuery'; import usePageFilters from 'sentry/utils/usePageFilters'; const ALL_INTERVAL_OPTIONS = [ {value: '1m', label: t('1 minute')}, {value: '5m', label: t('5 minutes')}, {value: '15m', label: t('15 minutes')}, {value: '30m', label: t('30 minutes')}, {value: '1h', label: t('1 hour')}, {value: '4h', label: t('4 hours')}, {value: '1d', label: t('1 day')}, {value: '1w', label: t('1 week')}, {value: '4w', label: t('1 month')}, ]; const minimumInterval = new GranularityLadder([ [SIXTY_DAYS, '1d'], [THIRTY_DAYS, '2h'], [TWO_WEEKS, '1h'], [ONE_WEEK, '30m'], [TWENTY_FOUR_HOURS, '5m'], [ONE_HOUR, '1m'], [0, '1m'], ]); const maximumInterval = new GranularityLadder([ [SIXTY_DAYS, '4w'], [THIRTY_DAYS, '1w'], [TWO_WEEKS, '1w'], [ONE_WEEK, '1d'], [TWENTY_FOUR_HOURS, '6h'], [ONE_HOUR, '15m'], [0, '1m'], ]); export function getIntervalOptionsForStatsPeriod(datetime: PageFilters['datetime']) { const diffInMinutes = getDiffInMinutes(datetime); const minimumOption = minimumInterval.getInterval(diffInMinutes); const minimumOptionInHours = parsePeriodToHours(minimumOption); const maximumOption = maximumInterval.getInterval(diffInMinutes); const maximumOptionInHours = parsePeriodToHours(maximumOption); return ALL_INTERVAL_OPTIONS.filter(option => { const optionInHours = parsePeriodToHours(option.value); return optionInHours >= minimumOptionInHours && optionInHours <= maximumOptionInHours; }); } export function validateInterval( interval: string, options: {label: string; value: string}[] ) { const isPeriod = !!parseStatsPeriod(interval); const currentIntervalValues = options.map(option => option.value); return isPeriod && currentIntervalValues.includes(interval) ? interval : // Take the 2nd most granular option if available options[1]?.value ?? options[0].value; } export function useMetricsIntervalParam() { const {datetime} = usePageFilters().selection; const {interval} = useLocationQuery({fields: {interval: decodeScalar}}); const updateQuery = useUpdateQuery(); const handleIntervalChange = useCallback( (newInterval: string) => { updateQuery({interval: newInterval}, {replace: true}); }, [updateQuery] ); const metricsIntervalOptions = useMetricsIntervalOptions({ interval, datetime, onIntervalChange: handleIntervalChange, }); useEffect(() => { if (interval !== metricsIntervalOptions.interval) { handleIntervalChange(metricsIntervalOptions.interval); } }, [interval, metricsIntervalOptions.interval, handleIntervalChange]); return metricsIntervalOptions; } export interface MetricsIntervalParamProps { datetime: PageFilters['datetime']; interval: string; onIntervalChange: (interval: string) => void; } export function useMetricsIntervalOptions({ interval, datetime, onIntervalChange, }: MetricsIntervalParamProps) { const currentIntervalOptions = useMemo( () => getIntervalOptionsForStatsPeriod(datetime), [datetime] ); const setInterval = useCallback( (newInterval: string) => { onIntervalChange(newInterval); }, [onIntervalChange] ); const validatedInterval = useMemo( () => validateInterval(interval, currentIntervalOptions), [interval, currentIntervalOptions] ); return { interval: validatedInterval, setInterval, currentIntervalOptions, }; }