123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218 |
- import styled from '@emotion/styled';
- import {getInterval} from 'sentry/components/charts/utils';
- import EmptyStateWarning from 'sentry/components/emptyStateWarning';
- import {OpsDot} from 'sentry/components/events/opsBreakdown';
- import LoadingContainer from 'sentry/components/loading/loadingContainer';
- import TextOverflow from 'sentry/components/textOverflow';
- import {t} from 'sentry/locale';
- import {space} from 'sentry/styles/space';
- import type {TableDataRow} from 'sentry/utils/discover/discoverQuery';
- import EventView from 'sentry/utils/discover/eventView';
- import {DiscoverDatasets} from 'sentry/utils/discover/types';
- import {formatVersion} from 'sentry/utils/formatters';
- import {decodeScalar} from 'sentry/utils/queryString';
- import {MutableSearch} from 'sentry/utils/tokenizeSearch';
- import {useLocation} from 'sentry/utils/useLocation';
- import usePageFilters from 'sentry/utils/usePageFilters';
- import {prepareQueryForLandingPage} from 'sentry/views/performance/data';
- import Breakdown from 'sentry/views/performance/mobile/appStarts/screens/breakdown';
- import {useTableQuery} from 'sentry/views/performance/mobile/screenload/screens/screensTable';
- import MiniChartPanel from 'sentry/views/starfish/components/miniChartPanel';
- import {useReleaseSelection} from 'sentry/views/starfish/queries/useReleases';
- import {formatVersionAndCenterTruncate} from 'sentry/views/starfish/utils/centerTruncate';
- import {STARFISH_CHART_INTERVAL_FIDELITY} from 'sentry/views/starfish/utils/constants';
- import {appendReleaseFilters} from 'sentry/views/starfish/utils/releaseComparison';
- const SYSTEM_COLOR = '#D6567F';
- const APPLICATION_COLOR = '#444674';
- const ANDROID_APPLICATION_SPAN_OPS = [
- 'contentprovider.load',
- 'application.load',
- 'activity.load',
- ];
- const IOS_APPLICATION_SPAN_DESCRIPTIONS = ['Initial Frame Render'];
- // Since we don't collect a tag that indicates whether a span is system or
- // application, we're going to use the span.op and span.description to
- // determine whether a span is system or application.
- function aggregateSystemApplicationBreakdown(data: TableDataRow[]) {
- return data.reduce((acc, row) => {
- const spanOp = row['span.op'] as string;
- const spanDescription = row['span.description'] as string;
- let type: 'system' | 'application' = 'system';
- if (
- ANDROID_APPLICATION_SPAN_OPS.includes(spanOp) ||
- IOS_APPLICATION_SPAN_DESCRIPTIONS.includes(spanDescription)
- ) {
- type = 'application';
- }
- acc[row.release] = {
- ...acc[row.release],
- [type]: (acc[row.release]?.[type] ?? 0) + (row['sum(span.self_time)'] ?? 0),
- };
- return acc;
- }, {});
- }
- function SystemApplicationBreakdown({additionalFilters}) {
- const pageFilter = usePageFilters();
- const location = useLocation();
- const {query: locationQuery} = location;
- const {
- primaryRelease,
- secondaryRelease,
- isLoading: isReleasesLoading,
- } = useReleaseSelection();
- const query = new MutableSearch([
- 'span.op:[app.start.warm,app.start.cold,contentprovider.load,application.load,activity.load]',
- '!span.description:"Cold Start"',
- '!span.description:"Warm Start"',
- ...(additionalFilters ?? []),
- ]);
- const searchQuery = decodeScalar(locationQuery.query, '');
- if (searchQuery) {
- query.addStringFilter(prepareQueryForLandingPage(searchQuery, false));
- }
- const queryString = appendReleaseFilters(query, primaryRelease, secondaryRelease);
- const {data, isLoading} = useTableQuery({
- eventView: EventView.fromNewQueryWithPageFilters(
- {
- name: '',
- fields: ['release', 'span.op', 'span.description', 'sum(span.self_time)'],
- topEvents: '2',
- query: queryString,
- dataset: DiscoverDatasets.SPANS_METRICS,
- version: 2,
- interval: getInterval(
- pageFilter.selection.datetime,
- STARFISH_CHART_INTERVAL_FIDELITY
- ),
- },
- pageFilter.selection
- ),
- enabled: !isReleasesLoading,
- referrer: 'api.starfish.mobile-startup-breakdown',
- initialData: {data: []},
- });
- if (isLoading) {
- return <LoadingContainer isLoading />;
- }
- if (!data) {
- return (
- <EmptyStateWarning small>
- <p>{t('There was no app start data found for these two releases')}</p>
- </EmptyStateWarning>
- );
- }
- const breakdownByReleaseData = aggregateSystemApplicationBreakdown(data.data);
- const breakdownGroups = [
- {
- color: SYSTEM_COLOR,
- key: 'system',
- name: t('System'),
- },
- {
- color: APPLICATION_COLOR,
- key: 'application',
- name: t('Application'),
- },
- ];
- return (
- <MiniChartPanel
- title={t('System v. Application')}
- subtitle={
- primaryRelease
- ? t(
- '%s v. %s',
- formatVersionAndCenterTruncate(primaryRelease, 12),
- secondaryRelease ? formatVersionAndCenterTruncate(secondaryRelease, 12) : ''
- )
- : ''
- }
- >
- <Legend>
- <LegendEntry>
- <StyledStartTypeDot style={{backgroundColor: SYSTEM_COLOR}} /> {t('System')}
- </LegendEntry>
- <LegendEntry>
- <StyledStartTypeDot style={{backgroundColor: APPLICATION_COLOR}} />
- {t('Application')}
- </LegendEntry>
- </Legend>
- <AppStartBreakdownContent>
- {primaryRelease && (
- <ReleaseAppStartBreakdown>
- <TextOverflow>{formatVersion(primaryRelease)}</TextOverflow>
- <Breakdown
- data-test-id="primary-release-breakdown"
- row={breakdownByReleaseData[primaryRelease]}
- breakdownGroups={breakdownGroups}
- />
- </ReleaseAppStartBreakdown>
- )}
- {secondaryRelease && (
- <ReleaseAppStartBreakdown>
- <TextOverflow>{formatVersion(secondaryRelease)}</TextOverflow>
- <Breakdown
- data-test-id="secondary-release-breakdown"
- row={breakdownByReleaseData[secondaryRelease]}
- breakdownGroups={breakdownGroups}
- />
- </ReleaseAppStartBreakdown>
- )}
- </AppStartBreakdownContent>
- </MiniChartPanel>
- );
- }
- export default SystemApplicationBreakdown;
- const ReleaseAppStartBreakdown = styled('div')`
- display: grid;
- grid-template-columns: 20% auto;
- gap: ${space(1)};
- color: ${p => p.theme.subText};
- `;
- const AppStartBreakdownContent = styled('div')`
- display: flex;
- flex-direction: column;
- gap: ${space(1)};
- margin-top: ${space(1)};
- `;
- const Legend = styled('div')`
- display: flex;
- gap: ${space(1.5)};
- position: absolute;
- top: ${space(1.5)};
- right: ${space(2)};
- `;
- const LegendEntry = styled('div')`
- display: flex;
- justify-content: center;
- align-items: center;
- font-size: ${p => p.theme.fontSizeSmall};
- `;
- const StyledStartTypeDot = styled(OpsDot)`
- position: relative;
- top: -1px;
- `;
|