123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167 |
- import {getInterval} from 'sentry/components/charts/utils';
- import {t} from 'sentry/locale';
- import {space} from 'sentry/styles/space';
- import type {Series, SeriesDataUnit} from 'sentry/types/echarts';
- import type {MultiSeriesEventsStats} from 'sentry/types/organization';
- import {defined} from 'sentry/utils';
- import {tooltipFormatterUsingAggregateOutputType} from 'sentry/utils/discover/charts';
- import EventView from 'sentry/utils/discover/eventView';
- import {DiscoverDatasets} from 'sentry/utils/discover/types';
- 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 {formatVersion} from 'sentry/utils/versions/formatVersion';
- import {
- PRIMARY_RELEASE_COLOR,
- SECONDARY_RELEASE_COLOR,
- } from 'sentry/views/insights/colors';
- import Chart, {ChartType} from 'sentry/views/insights/common/components/chart';
- import MiniChartPanel from 'sentry/views/insights/common/components/miniChartPanel';
- import {useReleaseSelection} from 'sentry/views/insights/common/queries/useReleases';
- import {formatVersionAndCenterTruncate} from 'sentry/views/insights/common/utils/centerTruncate';
- import {STARFISH_CHART_INTERVAL_FIDELITY} from 'sentry/views/insights/common/utils/constants';
- import {appendReleaseFilters} from 'sentry/views/insights/common/utils/releaseComparison';
- import {useEventsStatsQuery} from 'sentry/views/insights/common/utils/useEventsStatsQuery';
- import {COLD_START_TYPE} from 'sentry/views/insights/mobile/appStarts/components/startTypeSelector';
- import useCrossPlatformProject from 'sentry/views/insights/mobile/common/queries/useCrossPlatformProject';
- import {SpanMetricsField} from 'sentry/views/insights/types';
- const COLD_START_CONDITIONS = ['span.op:app.start.cold', 'span.description:"Cold Start"'];
- const WARM_START_CONDITIONS = ['span.op:app.start.warm', 'span.description:"Warm Start"'];
- export function transformData(data?: MultiSeriesEventsStats, primaryRelease?: string) {
- const transformedSeries: {[releaseName: string]: Series} = {};
- if (defined(data)) {
- Object.keys(data).forEach(releaseName => {
- transformedSeries[releaseName] = {
- seriesName: releaseName,
- data:
- data[releaseName]?.data?.map(datum => {
- return {
- name: datum[0] * 1000,
- value: datum[1][0].count,
- } as SeriesDataUnit;
- }) ?? [],
- ...(primaryRelease === releaseName
- ? {color: PRIMARY_RELEASE_COLOR}
- : {
- color: SECONDARY_RELEASE_COLOR,
- lineStyle: {type: 'dashed'},
- }),
- };
- });
- }
- return transformedSeries;
- }
- interface Props {
- chartHeight: number;
- additionalFilters?: string[];
- }
- function StartDurationWidget({additionalFilters, chartHeight}: Props) {
- const pageFilter = usePageFilters();
- const location = useLocation();
- const {
- primaryRelease,
- secondaryRelease,
- isLoading: isReleasesLoading,
- } = useReleaseSelection();
- const {isProjectCrossPlatform, selectedPlatform} = useCrossPlatformProject();
- const startType =
- decodeScalar(location.query[SpanMetricsField.APP_START_TYPE]) ?? COLD_START_TYPE;
- const query = new MutableSearch([
- ...(startType === COLD_START_TYPE ? COLD_START_CONDITIONS : WARM_START_CONDITIONS),
- ...(additionalFilters ?? []),
- ]);
- if (isProjectCrossPlatform) {
- query.addFilterValue('os.name', selectedPlatform);
- }
- const queryString = appendReleaseFilters(query, primaryRelease, secondaryRelease);
- const {
- data: series,
- isLoading: isSeriesLoading,
- error: seriesError,
- } = useEventsStatsQuery({
- eventView: EventView.fromNewQueryWithPageFilters(
- {
- name: '',
- topEvents: '2',
- fields: ['release', 'avg(span.duration)'],
- yAxis: ['avg(span.duration)'],
- 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-series',
- initialData: {},
- });
- // The expected response is a multi series response, but if there is no data
- // then we get an object representing a single series with all empty values
- // (i.e without being grouped by release)
- const hasReleaseData = series && !('data' in series);
- // Only transform the data is we know there's at least one release
- const transformedSeries = hasReleaseData
- ? Object.values(transformData(series, primaryRelease)).sort((releaseA, _releaseB) =>
- releaseA.seriesName === primaryRelease ? -1 : 1
- )
- : [];
- return (
- <MiniChartPanel
- title={
- startType === COLD_START_TYPE ? t('Average Cold Start') : t('Average Warm Start')
- }
- subtitle={
- primaryRelease
- ? t(
- '%s v. %s',
- formatVersionAndCenterTruncate(primaryRelease, 12),
- secondaryRelease ? formatVersionAndCenterTruncate(secondaryRelease, 12) : ''
- )
- : ''
- }
- >
- <Chart
- data={transformedSeries}
- height={chartHeight}
- loading={isSeriesLoading}
- grid={{
- left: '0',
- right: '0',
- top: space(2),
- bottom: '0',
- }}
- showLegend
- definedAxisTicks={2}
- type={ChartType.LINE}
- aggregateOutputFormat="duration"
- tooltipFormatterOptions={{
- valueFormatter: value =>
- tooltipFormatterUsingAggregateOutputType(value, 'duration'),
- nameFormatter: value => formatVersion(value),
- }}
- legendFormatter={value => formatVersion(value)}
- error={seriesError}
- />
- </MiniChartPanel>
- );
- }
- export default StartDurationWidget;
|