import {shouldFetchPreviousPeriod} from 'sentry/components/charts/utils'; import {normalizeDateTimeParams} from 'sentry/components/organizations/pageFilters/parse'; import {parseStatsPeriod} from 'sentry/components/timeRangeSelector/utils'; import {t} from 'sentry/locale'; import type {PageFilters} from 'sentry/types/core'; import type {Organization} from 'sentry/types/organization'; import {getPeriod} from 'sentry/utils/duration/getPeriod'; import {useApiQuery} from 'sentry/utils/queryClient'; import {BigNumberWidget} from 'sentry/views/dashboards/widgets/bigNumberWidget/bigNumberWidget'; import {WidgetFrame} from 'sentry/views/dashboards/widgets/common/widgetFrame'; import MissingReleasesButtons from '../missingFeatureButtons/missingReleasesButtons'; const API_LIMIT = 1000; type Release = {date: string; version: string}; const useReleaseCount = (props: Props) => { const {organization, selection, isProjectStabilized, query} = props; const isEnabled = isProjectStabilized; const {projects, environments, datetime} = selection; const {period} = datetime; const {start: previousStart} = parseStatsPeriod( getPeriod({period, start: undefined, end: undefined}, {shouldDoublePeriod: true}) .statsPeriod! ); const {start: previousEnd} = parseStatsPeriod( getPeriod({period, start: undefined, end: undefined}, {shouldDoublePeriod: false}) .statsPeriod! ); const commonQuery = { environment: environments, project: projects[0], query, }; const currentQuery = useApiQuery( [ `/organizations/${organization.slug}/releases/stats/`, { query: { ...commonQuery, ...normalizeDateTimeParams(datetime), }, }, ], {staleTime: 0, enabled: isEnabled} ); const isPreviousPeriodEnabled = shouldFetchPreviousPeriod({ start: datetime.start, end: datetime.end, period: datetime.period, }); const previousQuery = useApiQuery( [ `/organizations/${organization.slug}/releases/stats/`, { query: { ...commonQuery, start: previousStart, end: previousEnd, }, }, ], { staleTime: 0, enabled: isEnabled && isPreviousPeriodEnabled, } ); const allReleases = [...(currentQuery.data ?? []), ...(previousQuery.data ?? [])]; const isAllTimePeriodEnabled = !currentQuery.isPending && !currentQuery.error && !previousQuery.isPending && !previousQuery.error && allReleases.length === 0; const allTimeQuery = useApiQuery( [ `/organizations/${organization.slug}/releases/stats/`, { query: { ...commonQuery, statsPeriod: '90d', per_page: 1, }, }, ], { staleTime: 0, enabled: isEnabled && isAllTimePeriodEnabled, } ); return { data: currentQuery.data, previousData: previousQuery.data, allTimeData: allTimeQuery.data, isLoading: currentQuery.isPending || (previousQuery.isPending && isPreviousPeriodEnabled) || (allTimeQuery.isPending && isAllTimePeriodEnabled), error: currentQuery.error || previousQuery.error || allTimeQuery.error, refetch: () => { currentQuery.refetch(); previousQuery.refetch(); allTimeQuery.refetch(); }, }; }; type Props = { isProjectStabilized: boolean; organization: Organization; selection: PageFilters; query?: string; }; function ProjectVelocityScoreCard(props: Props) { const {organization} = props; const { data: currentReleases, previousData: previousReleases, allTimeData: allTimeReleases, isLoading, error, refetch, } = useReleaseCount(props); const noReleaseEver = [...(allTimeReleases ?? []), ...(previousReleases ?? []), ...(allTimeReleases ?? [])] .length === 0; const cardTitle = t('Number of Releases'); const cardHelp = t('The number of releases for this project.'); if (!isLoading && noReleaseEver) { return ( ); } return ( ); } export default ProjectVelocityScoreCard;