123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170 |
- import styled from '@emotion/styled';
- import {Location} from 'history';
- import pick from 'lodash/pick';
- import moment from 'moment';
- import AsyncComponent from 'sentry/components/asyncComponent';
- import Count from 'sentry/components/count';
- import {normalizeDateTimeParams} from 'sentry/components/organizations/pageFilters/parse';
- import * as SidebarSection from 'sentry/components/sidebarSection';
- import {URL_PARAM} from 'sentry/constants/pageFilters';
- import {t, tn} from 'sentry/locale';
- import space from 'sentry/styles/space';
- import {CrashFreeTimeBreakdown, Organization} from 'sentry/types';
- import {defined} from 'sentry/utils';
- import {displayCrashFreePercent} from '../../../utils';
- type Props = AsyncComponent['props'] & {
- location: Location;
- organization: Organization;
- projectSlug: string;
- version: string;
- };
- type State = AsyncComponent['state'] & {
- releaseStats?: {usersBreakdown: CrashFreeTimeBreakdown} | null;
- };
- class TotalCrashFreeUsers extends AsyncComponent<Props, State> {
- shouldReload = true;
- getEndpoints(): ReturnType<AsyncComponent['getEndpoints']> {
- const {location, organization, projectSlug, version} = this.props;
- return [
- [
- 'releaseStats',
- `/projects/${organization.slug}/${projectSlug}/releases/${encodeURIComponent(
- version
- )}/stats/`,
- {
- query: {
- ...normalizeDateTimeParams(
- pick(location.query, [URL_PARAM.PROJECT, URL_PARAM.ENVIRONMENT])
- ),
- type: 'sessions',
- },
- },
- ],
- ];
- }
- componentDidUpdate(prevProps: Props) {
- if (prevProps.version !== this.props.version) {
- this.remountComponent();
- }
- }
- renderLoading() {
- return this.renderBody();
- }
- renderBody() {
- const crashFreeTimeBreakdown = this.state.releaseStats?.usersBreakdown;
- if (!crashFreeTimeBreakdown?.length) {
- return null;
- }
- const timeline = crashFreeTimeBreakdown
- .map(({date, crashFreeUsers, totalUsers}, index, data) => {
- // count number of crash free users from knowing percent and total
- const crashFreeUserCount = Math.round(((crashFreeUsers ?? 0) * totalUsers) / 100);
- // first item of timeline is release creation date, then we want to have relative date label
- const dateLabel =
- index === 0
- ? t('Release created')
- : `${moment(data[0].date).from(date, true)} ${t('later')}`;
- return {date: moment(date), dateLabel, crashFreeUsers, crashFreeUserCount};
- })
- // remove those timeframes that are in the future
- .filter(item => item.date.isBefore())
- // we want timeline to go from bottom to up
- .reverse();
- if (!timeline.length) {
- return null;
- }
- return (
- <SidebarSection.Wrap>
- <SidebarSection.Title>{t('Total Crash Free Users')}</SidebarSection.Title>
- <SidebarSection.Content>
- <Timeline>
- {timeline.map(row => (
- <Row key={row.date.toString()}>
- <InnerRow>
- <Text bold>{row.date.format('MMMM D')}</Text>
- <Text bold right>
- <Count value={row.crashFreeUserCount} />{' '}
- {tn('user', 'users', row.crashFreeUserCount)}
- </Text>
- </InnerRow>
- <InnerRow>
- <Text>{row.dateLabel}</Text>
- <Percent right>
- {defined(row.crashFreeUsers)
- ? displayCrashFreePercent(row.crashFreeUsers)
- : '-'}
- </Percent>
- </InnerRow>
- </Row>
- ))}
- </Timeline>
- </SidebarSection.Content>
- </SidebarSection.Wrap>
- );
- }
- }
- const Timeline = styled('div')`
- font-size: ${p => p.theme.fontSizeMedium};
- line-height: 1.2;
- `;
- const DOT_SIZE = 10;
- const Row = styled('div')`
- border-left: 1px solid ${p => p.theme.border};
- padding-left: ${space(2)};
- padding-bottom: ${space(1)};
- margin-left: ${space(1)};
- position: relative;
- &:before {
- content: '';
- width: ${DOT_SIZE}px;
- height: ${DOT_SIZE}px;
- border-radius: 100%;
- background-color: ${p => p.theme.purple300};
- position: absolute;
- top: 0;
- left: -${Math.floor(DOT_SIZE / 2)}px;
- }
- &:last-child {
- border-left: 0;
- }
- `;
- const InnerRow = styled('div')`
- display: grid;
- grid-column-gap: ${space(2)};
- grid-auto-flow: column;
- grid-auto-columns: 1fr;
- padding-bottom: ${space(0.5)};
- `;
- const Text = styled('div')<{bold?: boolean; right?: boolean}>`
- text-align: ${p => (p.right ? 'right' : 'left')};
- color: ${p => (p.bold ? p.theme.textColor : p.theme.gray300)};
- padding-bottom: ${space(0.25)};
- ${p => p.theme.overflowEllipsis};
- `;
- const Percent = styled(Text)`
- font-variant-numeric: tabular-nums;
- `;
- export default TotalCrashFreeUsers;
|