123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402 |
- import {browserHistory} from 'react-router';
- import {useTheme} from '@emotion/react';
- import styled from '@emotion/styled';
- import color from 'color';
- import ChartZoom from 'sentry/components/charts/chartZoom';
- import MarkPoint from 'sentry/components/charts/components/markPoint';
- import ErrorPanel from 'sentry/components/charts/errorPanel';
- import EventsRequest from 'sentry/components/charts/eventsRequest';
- import {LineChart, LineChartProps} from 'sentry/components/charts/lineChart';
- import {SectionHeading} from 'sentry/components/charts/styles';
- import TransitionChart from 'sentry/components/charts/transitionChart';
- import TransparentLoadingMask from 'sentry/components/charts/transparentLoadingMask';
- import {getInterval} from 'sentry/components/charts/utils';
- import {normalizeDateTimeParams} from 'sentry/components/organizations/pageFilters/parse';
- import Placeholder from 'sentry/components/placeholder';
- import QuestionTooltip from 'sentry/components/questionTooltip';
- import {IconWarning} from 'sentry/icons';
- import {t} from 'sentry/locale';
- import {Organization} from 'sentry/types';
- import {getUtcToLocalDateObject} from 'sentry/utils/dates';
- import {tooltipFormatter} from 'sentry/utils/discover/charts';
- import EventView from 'sentry/utils/discover/eventView';
- import {aggregateOutputType} from 'sentry/utils/discover/fields';
- import {QueryError} from 'sentry/utils/discover/genericDiscoverQuery';
- import {formatFloat, formatPercentage} from 'sentry/utils/formatters';
- import getDynamicText from 'sentry/utils/getDynamicText';
- import AnomaliesQuery from 'sentry/utils/performance/anomalies/anomaliesQuery';
- import {useMEPSettingContext} from 'sentry/utils/performance/contexts/metricsEnhancedSetting';
- import {decodeScalar} from 'sentry/utils/queryString';
- import useApi from 'sentry/utils/useApi';
- import {useLocation} from 'sentry/utils/useLocation';
- import useRouter from 'sentry/utils/useRouter';
- import {normalizeUrl} from 'sentry/utils/withDomainRequired';
- import {getTermHelp, PERFORMANCE_TERM} from 'sentry/views/performance/data';
- import {getTransactionMEPParamsIfApplicable} from 'sentry/views/performance/transactionSummary/transactionOverview/utils';
- import {
- anomaliesRouteWithQuery,
- ANOMALY_FLAG,
- anomalyToColor,
- } from '../transactionAnomalies/utils';
- type ContainerProps = {
- error: QueryError | null;
- eventView: EventView;
- isLoading: boolean;
- organization: Organization;
- totals: Record<string, number> | null;
- transactionName: string;
- };
- type Props = Pick<ContainerProps, 'organization' | 'isLoading' | 'error' | 'totals'> & {
- chartData: {
- chartOptions: Omit<LineChartProps, 'series'>;
- errored: boolean;
- loading: boolean;
- reloading: boolean;
- series: LineChartProps['series'];
- };
- eventView: EventView;
- transactionName: string;
- utc: boolean;
- end?: Date;
- start?: Date;
- statsPeriod?: string | null;
- };
- function SidebarCharts({
- organization,
- isLoading,
- error,
- totals,
- start,
- end,
- utc,
- statsPeriod,
- chartData,
- eventView,
- transactionName,
- }: Props) {
- const location = useLocation();
- const router = useRouter();
- const theme = useTheme();
- return (
- <RelativeBox>
- <ChartLabel top="0px">
- <ChartTitle>
- {t('Apdex')}
- <QuestionTooltip
- position="top"
- title={getTermHelp(organization, PERFORMANCE_TERM.APDEX)}
- size="sm"
- />
- </ChartTitle>
- <ChartSummaryValue
- data-test-id="apdex-summary-value"
- isLoading={isLoading}
- error={error}
- value={totals ? formatFloat(totals['apdex()'], 4) : null}
- />
- </ChartLabel>
- <ChartLabel top="160px">
- <ChartTitle>
- {t('Failure Rate')}
- <QuestionTooltip
- position="top"
- title={getTermHelp(organization, PERFORMANCE_TERM.FAILURE_RATE)}
- size="sm"
- />
- </ChartTitle>
- <ChartSummaryValue
- data-test-id="failure-rate-summary-value"
- isLoading={isLoading}
- error={error}
- value={totals ? formatPercentage(totals['failure_rate()']) : null}
- />
- </ChartLabel>
- <AnomaliesQuery
- location={location}
- organization={organization}
- eventView={eventView}
- >
- {results => (
- <ChartZoom
- router={router}
- period={statsPeriod}
- start={start}
- end={end}
- utc={utc}
- xAxisIndex={[0, 1, 2]}
- >
- {zoomRenderProps => {
- const {errored, loading, reloading, chartOptions, series} = chartData;
- if (errored) {
- return (
- <ErrorPanel height="300px">
- <IconWarning color="gray300" size="lg" />
- </ErrorPanel>
- );
- }
- if (organization.features.includes(ANOMALY_FLAG)) {
- const epmSeries = series.find(
- s => s.seriesName.includes('epm') || s.seriesName.includes('tpm')
- );
- if (epmSeries && results.data) {
- epmSeries.markPoint = MarkPoint({
- data: results.data.anomalies.map(a => ({
- name: a.id,
- yAxis: epmSeries.data.find(
- ({name}) => (name as number) > (a.end + a.start) / 2
- )?.value,
- // TODO: the above is O(n*m), remove after we change the api to include the midpoint of y.
- xAxis: a.start,
- itemStyle: {
- borderColor: color(anomalyToColor(a.confidence, theme)).string(),
- color: color(anomalyToColor(a.confidence, theme))
- .alpha(0.2)
- .rgb()
- .string(),
- },
- onClick: () => {
- const target = anomaliesRouteWithQuery({
- orgSlug: organization.slug,
- query: location.query,
- projectID: decodeScalar(location.query.project),
- transaction: transactionName,
- });
- browserHistory.push(normalizeUrl(target));
- },
- })),
- symbol: 'circle',
- symbolSize: 16,
- });
- }
- }
- return (
- <TransitionChart loading={loading} reloading={reloading} height="580px">
- <TransparentLoadingMask visible={reloading} />
- {getDynamicText({
- value: (
- <LineChart {...zoomRenderProps} {...chartOptions} series={series} />
- ),
- fixed: <Placeholder height="300px" testId="skeleton-ui" />,
- })}
- </TransitionChart>
- );
- }}
- </ChartZoom>
- )}
- </AnomaliesQuery>
- </RelativeBox>
- );
- }
- function SidebarChartsContainer({
- eventView,
- organization,
- isLoading,
- error,
- totals,
- transactionName,
- }: ContainerProps) {
- const location = useLocation();
- const router = useRouter();
- const api = useApi();
- const theme = useTheme();
- const colors = theme.charts.getColorPalette(2);
- const statsPeriod = eventView.statsPeriod;
- const start = eventView.start ? getUtcToLocalDateObject(eventView.start) : undefined;
- const end = eventView.end ? getUtcToLocalDateObject(eventView.end) : undefined;
- const project = eventView.project;
- const environment = eventView.environment;
- const query = eventView.query;
- const utc = normalizeDateTimeParams(location.query).utc === 'true';
- const mepSetting = useMEPSettingContext();
- const queryExtras = getTransactionMEPParamsIfApplicable(mepSetting, organization);
- const axisLineConfig = {
- scale: true,
- axisLine: {
- show: false,
- },
- axisTick: {
- show: false,
- },
- splitLine: {
- show: false,
- },
- };
- const chartOptions: Omit<LineChartProps, 'series'> = {
- height: 300,
- grid: [
- {
- top: '60px',
- left: '10px',
- right: '10px',
- height: '100px',
- },
- {
- top: '220px',
- left: '10px',
- right: '10px',
- height: '100px',
- },
- ],
- axisPointer: {
- // Link each x-axis together.
- link: [{xAxisIndex: [0, 1]}],
- },
- xAxes: Array.from(new Array(2)).map((_i, index) => ({
- gridIndex: index,
- type: 'time',
- show: false,
- })),
- yAxes: [
- {
- // apdex
- gridIndex: 0,
- interval: 0.2,
- axisLabel: {
- formatter: (value: number) => `${formatFloat(value, 1)}`,
- color: theme.chartLabel,
- },
- ...axisLineConfig,
- },
- {
- // failure rate
- gridIndex: 1,
- splitNumber: 4,
- interval: 0.5,
- max: 1.0,
- axisLabel: {
- formatter: (value: number) => formatPercentage(value, 0),
- color: theme.chartLabel,
- },
- ...axisLineConfig,
- },
- ],
- utc,
- isGroupedByDate: true,
- showTimeInTooltip: true,
- colors: [colors[0], colors[1]],
- tooltip: {
- trigger: 'axis',
- truncate: 80,
- valueFormatter: (value, label) =>
- tooltipFormatter(value, aggregateOutputType(label)),
- nameFormatter(value: string) {
- return value === 'epm()' ? 'tpm()' : value;
- },
- },
- };
- const requestCommonProps = {
- api,
- start,
- end,
- period: statsPeriod,
- project,
- environment,
- query,
- };
- const contentCommonProps = {
- organization,
- router,
- error,
- isLoading,
- start,
- end,
- utc,
- totals,
- };
- const datetimeSelection = {
- start: start || null,
- end: end || null,
- period: statsPeriod,
- };
- return (
- <EventsRequest
- {...requestCommonProps}
- organization={organization}
- interval={getInterval(datetimeSelection)}
- showLoading={false}
- includePrevious={false}
- yAxis={['apdex()', 'failure_rate()']}
- partial
- referrer="api.performance.transaction-summary.sidebar-chart"
- queryExtras={queryExtras}
- >
- {({results, errored, loading, reloading}) => {
- const series = results
- ? results.map((v, i: number) => ({
- ...v,
- yAxisIndex: i,
- xAxisIndex: i,
- }))
- : [];
- return (
- <SidebarCharts
- {...contentCommonProps}
- transactionName={transactionName}
- eventView={eventView}
- chartData={{series, errored, loading, reloading, chartOptions}}
- />
- );
- }}
- </EventsRequest>
- );
- }
- type ChartValueProps = {
- 'data-test-id': string;
- error: QueryError | null;
- isLoading: boolean;
- value: React.ReactNode;
- };
- function ChartSummaryValue({error, isLoading, value, ...props}: ChartValueProps) {
- if (error) {
- return <div {...props}>{'\u2014'}</div>;
- }
- if (isLoading) {
- return <Placeholder height="24px" {...props} />;
- }
- return <ChartValue {...props}>{value}</ChartValue>;
- }
- const RelativeBox = styled('div')`
- position: relative;
- `;
- const ChartTitle = styled(SectionHeading)`
- margin: 0;
- `;
- const ChartLabel = styled('div')<{top: string}>`
- position: absolute;
- top: ${p => p.top};
- z-index: 1;
- `;
- const ChartValue = styled('div')`
- font-size: ${p => p.theme.fontSizeExtraLarge};
- `;
- export default SidebarChartsContainer;
|