import {Fragment} from 'react'; import {fetchAnyReleaseExistence} from 'sentry/actionCreators/projects'; import AsyncComponent from 'sentry/components/asyncComponent'; import {shouldFetchPreviousPeriod} from 'sentry/components/charts/utils'; import {normalizeDateTimeParams} from 'sentry/components/organizations/pageFilters/parse'; import {parseStatsPeriod} from 'sentry/components/organizations/timeRangeSelector/utils'; import ScoreCard from 'sentry/components/scoreCard'; import {IconArrow} from 'sentry/icons'; import {t} from 'sentry/locale'; import {Organization, PageFilters} from 'sentry/types'; import {defined} from 'sentry/utils'; import {getPeriod} from 'sentry/utils/getPeriod'; import MissingReleasesButtons from '../missingFeatureButtons/missingReleasesButtons'; const API_LIMIT = 1000; type Release = {date: string; version: string}; type Props = AsyncComponent['props'] & { isProjectStabilized: boolean; organization: Organization; selection: PageFilters; query?: string; }; type State = AsyncComponent['state'] & { currentReleases: Release[] | null; noReleaseEver: boolean; previousReleases: Release[] | null; }; class ProjectVelocityScoreCard extends AsyncComponent { shouldRenderBadRequests = true; getDefaultState() { return { ...super.getDefaultState(), currentReleases: null, previousReleases: null, noReleaseEver: false, }; } getEndpoints() { const {organization, selection, isProjectStabilized, query} = this.props; if (!isProjectStabilized) { return []; } const {projects, environments, datetime} = selection; const {period} = datetime; const commonQuery = { environment: environments, project: projects[0], query, }; const endpoints: ReturnType = [ [ 'currentReleases', `/organizations/${organization.slug}/releases/stats/`, { includeAllArgs: true, method: 'GET', query: { ...commonQuery, ...normalizeDateTimeParams(datetime), }, }, ], ]; if ( shouldFetchPreviousPeriod({ start: datetime.start, end: datetime.end, period: datetime.period, }) ) { 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! ); endpoints.push([ 'previousReleases', `/organizations/${organization.slug}/releases/stats/`, { query: { ...commonQuery, start: previousStart, end: previousEnd, }, }, ]); } return endpoints; } /** * If our releases are empty, determine if we had a release in the last 90 days (empty message differs then) */ async onLoadAllEndpointsSuccess() { const {currentReleases, previousReleases} = this.state; const {organization, selection, isProjectStabilized} = this.props; if (!isProjectStabilized) { return; } if ([...(currentReleases ?? []), ...(previousReleases ?? [])].length !== 0) { this.setState({noReleaseEver: false}); return; } this.setState({loading: true}); const hasOlderReleases = await fetchAnyReleaseExistence( this.api, organization.slug, selection.projects[0] ); this.setState({noReleaseEver: !hasOlderReleases, loading: false}); } get cardTitle() { return t('Number of Releases'); } get cardHelp() { return this.trend ? t( 'The number of releases for this project and how it has changed since the last period.' ) : t('The number of releases for this project.'); } get trend() { const {currentReleases, previousReleases} = this.state; if (!defined(currentReleases) || !defined(previousReleases)) { return null; } return currentReleases.length - previousReleases.length; } get trendStatus(): React.ComponentProps['trendStatus'] { if (!this.trend) { return undefined; } return this.trend > 0 ? 'good' : 'bad'; } componentDidUpdate(prevProps: Props) { const {selection, isProjectStabilized, query} = this.props; if ( prevProps.selection !== selection || prevProps.isProjectStabilized !== isProjectStabilized || prevProps.query !== query ) { this.remountComponent(); } } renderLoading() { return this.renderBody(); } renderMissingFeatureCard() { const {organization} = this.props; return ( } /> ); } renderScore() { const {currentReleases, loading} = this.state; if (loading || !defined(currentReleases)) { return '\u2014'; } return currentReleases.length === API_LIMIT ? `${API_LIMIT - 1}+` : currentReleases.length; } renderTrend() { const {loading, currentReleases} = this.state; if (loading || !defined(this.trend) || currentReleases?.length === API_LIMIT) { return null; } return ( {this.trend >= 0 ? ( ) : ( )} {Math.abs(this.trend)} ); } renderBody() { const {noReleaseEver} = this.state; if (noReleaseEver) { return this.renderMissingFeatureCard(); } return ( ); } } export default ProjectVelocityScoreCard;