import {Fragment} from 'react'; import {browserHistory} from 'react-router'; import styled from '@emotion/styled'; import {Location} from 'history'; import omit from 'lodash/omit'; import DatePageFilter from 'sentry/components/datePageFilter'; import TransactionsList, { DropdownOption, } from 'sentry/components/discover/transactionsList'; import EnvironmentPageFilter from 'sentry/components/environmentPageFilter'; import SearchBar from 'sentry/components/events/searchBar'; import * as Layout from 'sentry/components/layouts/thirds'; import PageFilterBar from 'sentry/components/organizations/pageFilterBar'; import {normalizeDateTimeParams} from 'sentry/components/organizations/pageFilters/parse'; import {SuspectFunctionsTable} from 'sentry/components/profiling/suspectFunctions/suspectFunctionsTable'; import {ActionBarItem} from 'sentry/components/smartSearchBar'; import {Tooltip} from 'sentry/components/tooltip'; import {MAX_QUERY_LENGTH} from 'sentry/constants'; import {IconWarning} from 'sentry/icons'; import {t} from 'sentry/locale'; import {space} from 'sentry/styles/space'; import {Organization, Project} from 'sentry/types'; import {defined, generateQueryWithTag} from 'sentry/utils'; import {trackAnalytics} from 'sentry/utils/analytics'; import EventView from 'sentry/utils/discover/eventView'; import { formatTagKey, isRelativeSpanOperationBreakdownField, SPAN_OP_BREAKDOWN_FIELDS, SPAN_OP_RELATIVE_BREAKDOWN_FIELD, } from 'sentry/utils/discover/fields'; import {QueryError} from 'sentry/utils/discover/genericDiscoverQuery'; import { MetricsEnhancedPerformanceDataContext, useMEPDataContext, } from 'sentry/utils/performance/contexts/metricsEnhancedPerformanceDataContext'; import {decodeScalar} from 'sentry/utils/queryString'; import projectSupportsReplay from 'sentry/utils/replays/projectSupportsReplay'; import {useRoutes} from 'sentry/utils/useRoutes'; import withProjects from 'sentry/utils/withProjects'; import {Actions, updateQuery} from 'sentry/views/discover/table/cellAction'; import {TableColumn} from 'sentry/views/discover/table/types'; import Tags from 'sentry/views/discover/tags'; import {TransactionPercentage} from 'sentry/views/performance/transactionSummary/transactionOverview/transactionPercentage'; import {canUseTransactionMetricsData} from 'sentry/views/performance/transactionSummary/transactionOverview/utils'; import { PERCENTILE as VITAL_PERCENTILE, VITAL_GROUPS, } from 'sentry/views/performance/transactionSummary/transactionVitals/constants'; import {isSummaryViewFrontend, isSummaryViewFrontendPageLoad} from '../../utils'; import Filter, { decodeFilterFromLocation, filterToField, filterToSearchConditions, SpanOperationBreakdownFilter, } from '../filter'; import { generateProfileLink, generateReplayLink, generateTraceLink, generateTransactionLink, normalizeSearchConditions, SidebarSpacer, TransactionFilterOptions, } from '../utils'; import TransactionSummaryCharts from './charts'; import {PerformanceAtScaleContextProvider} from './performanceAtScaleContext'; import RelatedIssues from './relatedIssues'; import SidebarCharts from './sidebarCharts'; import StatusBreakdown from './statusBreakdown'; import SuspectSpans from './suspectSpans'; import {TagExplorer} from './tagExplorer'; import UserStats from './userStats'; type Props = { error: QueryError | null; eventView: EventView; isLoading: boolean; location: Location; onChangeFilter: (newFilter: SpanOperationBreakdownFilter) => void; organization: Organization; projectId: string; projects: Project[]; spanOperationBreakdownFilter: SpanOperationBreakdownFilter; totalValues: Record | null; transactionName: string; unfilteredTotalValues?: Record | null; }; function SummaryContent({ eventView, location, totalValues, spanOperationBreakdownFilter, organization, projects, isLoading, error, projectId, transactionName, onChangeFilter, unfilteredTotalValues, }: Props) { const routes = useRoutes(); const mepDataContext = useMEPDataContext(); function handleSearch(query: string) { const queryParams = normalizeDateTimeParams({ ...(location.query || {}), query, }); // do not propagate pagination when making a new search const searchQueryParams = omit(queryParams, 'cursor'); browserHistory.push({ pathname: location.pathname, query: searchQueryParams, }); } function generateTagUrl(key: string, value: string) { const query = generateQueryWithTag(location.query, {key: formatTagKey(key), value}); return { ...location, query, }; } function handleCellAction(column: TableColumn) { return (action: Actions, value: React.ReactText) => { const searchConditions = normalizeSearchConditions(eventView.query); updateQuery(searchConditions, action, column, value); browserHistory.push({ pathname: location.pathname, query: { ...location.query, cursor: undefined, query: searchConditions.formatString(), }, }); }; } function handleTransactionsListSortChange(value: string) { const target = { pathname: location.pathname, query: {...location.query, showTransactions: value, transactionCursor: undefined}, }; browserHistory.push(target); } function handleAllEventsViewClick() { trackAnalytics('performance_views.summary.view_in_transaction_events', { organization, }); } function generateEventView( transactionsListEventView: EventView, transactionsListTitles: string[] ) { const {selected} = getTransactionsListSort(location, { p95: totalValues?.['p95()'] ?? 0, spanOperationBreakdownFilter, }); const sortedEventView = transactionsListEventView.withSorts([selected.sort]); if (spanOperationBreakdownFilter === SpanOperationBreakdownFilter.None) { const fields = [ // Remove the extra field columns ...sortedEventView.fields.slice(0, transactionsListTitles.length), ]; // omit "Operation Duration" column sortedEventView.fields = fields.filter(({field}) => { return !isRelativeSpanOperationBreakdownField(field); }); } return sortedEventView; } function generateActionBarItems( _org: Organization, _location: Location, _mepDataContext: MetricsEnhancedPerformanceDataContext ) { let items: ActionBarItem[] | undefined = undefined; if (!canUseTransactionMetricsData(_org, _mepDataContext)) { items = [ { key: 'alert', makeAction: () => ({ Button: () => ( ), menuItem: { key: 'alert', }, }), }, ]; } return items; } const hasPerformanceChartInterpolation = organization.features.includes( 'performance-chart-interpolation' ); const query = decodeScalar(location.query.query, ''); const totalCount = totalValues === null ? null : totalValues['count()']; // NOTE: This is not a robust check for whether or not a transaction is a front end // transaction, however it will suffice for now. const hasWebVitals = isSummaryViewFrontendPageLoad(eventView, projects) || (totalValues !== null && VITAL_GROUPS.some(group => group.vitals.some(vital => { const functionName = `percentile(${vital},${VITAL_PERCENTILE})`; const field = functionName; return Number.isFinite(totalValues[field]) && totalValues[field] !== 0; }) )); const isFrontendView = isSummaryViewFrontend(eventView, projects); const transactionsListTitles = [ t('event id'), t('user'), t('total duration'), t('trace id'), t('timestamp'), ]; const project = projects.find(p => p.id === projectId); let transactionsListEventView = eventView.clone(); const fields = [...transactionsListEventView.fields]; if ( organization.features.includes('session-replay') && project && projectSupportsReplay(project) ) { transactionsListTitles.push(t('replay')); fields.push({field: 'replayId'}); } if ( organization.features.includes('profiling') && project && // only show for projects that already sent a profile // once we have a more compact design we will show this for // projects that support profiling as well project.hasProfiles ) { transactionsListTitles.push(t('profile')); fields.push({field: 'profile.id'}); } // update search conditions const spanOperationBreakdownConditions = filterToSearchConditions( spanOperationBreakdownFilter, location ); if (spanOperationBreakdownConditions) { eventView = eventView.clone(); eventView.query = `${eventView.query} ${spanOperationBreakdownConditions}`.trim(); transactionsListEventView = eventView.clone(); } // update header titles of transactions list const operationDurationTableTitle = spanOperationBreakdownFilter === SpanOperationBreakdownFilter.None ? t('operation duration') : `${spanOperationBreakdownFilter} duration`; // add ops breakdown duration column as the 3rd column transactionsListTitles.splice(2, 0, operationDurationTableTitle); // span_ops_breakdown.relative is a preserved name and a marker for the associated // field renderer to be used to generate the relative ops breakdown let durationField = SPAN_OP_RELATIVE_BREAKDOWN_FIELD; if (spanOperationBreakdownFilter !== SpanOperationBreakdownFilter.None) { durationField = filterToField(spanOperationBreakdownFilter)!; } // add ops breakdown duration column as the 3rd column fields.splice(2, 0, {field: durationField}); if (spanOperationBreakdownFilter === SpanOperationBreakdownFilter.None) { fields.push( ...SPAN_OP_BREAKDOWN_FIELDS.map(field => { return {field}; }) ); } transactionsListEventView.fields = fields; const openAllEventsProps = { generatePerformanceTransactionEventsView: () => { const performanceTransactionEventsView = generateEventView( transactionsListEventView, transactionsListTitles ); performanceTransactionEventsView.query = query; return performanceTransactionEventsView; }, handleOpenAllEventsClick: handleAllEventsViewClick, }; return ( {!isFrontendView && ( )} ); } function getFilterOptions({ p95, spanOperationBreakdownFilter, }: { p95: number; spanOperationBreakdownFilter: SpanOperationBreakdownFilter; }): DropdownOption[] { if (spanOperationBreakdownFilter === SpanOperationBreakdownFilter.None) { return [ { sort: {kind: 'asc', field: 'transaction.duration'}, value: TransactionFilterOptions.FASTEST, label: t('Fastest Transactions'), }, { query: [['transaction.duration', `<=${p95.toFixed(0)}`]], sort: {kind: 'desc', field: 'transaction.duration'}, value: TransactionFilterOptions.SLOW, label: t('Slow Transactions (p95)'), }, { sort: {kind: 'desc', field: 'transaction.duration'}, value: TransactionFilterOptions.OUTLIER, label: t('Outlier Transactions (p100)'), }, { sort: {kind: 'desc', field: 'timestamp'}, value: TransactionFilterOptions.RECENT, label: t('Recent Transactions'), }, ]; } const field = filterToField(spanOperationBreakdownFilter)!; const operationName = spanOperationBreakdownFilter; return [ { sort: {kind: 'asc', field}, value: TransactionFilterOptions.FASTEST, label: t('Fastest %s Operations', operationName), }, { query: [['transaction.duration', `<=${p95.toFixed(0)}`]], sort: {kind: 'desc', field}, value: TransactionFilterOptions.SLOW, label: t('Slow %s Operations (p95)', operationName), }, { sort: {kind: 'desc', field}, value: TransactionFilterOptions.OUTLIER, label: t('Outlier %s Operations (p100)', operationName), }, { sort: {kind: 'desc', field: 'timestamp'}, value: TransactionFilterOptions.RECENT, label: t('Recent Transactions'), }, ]; } function getTransactionsListSort( location: Location, options: {p95: number; spanOperationBreakdownFilter: SpanOperationBreakdownFilter} ): {options: DropdownOption[]; selected: DropdownOption} { const sortOptions = getFilterOptions(options); const urlParam = decodeScalar( location.query.showTransactions, TransactionFilterOptions.SLOW ); const selectedSort = sortOptions.find(opt => opt.value === urlParam) || sortOptions[0]; return {selected: selectedSort, options: sortOptions}; } const FilterActions = styled('div')` display: grid; gap: ${space(2)}; margin-bottom: ${space(2)}; @media (min-width: ${p => p.theme.breakpoints.small}) { grid-template-columns: repeat(2, min-content); } @media (min-width: ${p => p.theme.breakpoints.xlarge}) { grid-template-columns: auto auto 1fr; } `; const StyledSearchBar = styled(SearchBar)` @media (min-width: ${p => p.theme.breakpoints.small}) { order: 1; grid-column: 1/4; } @media (min-width: ${p => p.theme.breakpoints.xlarge}) { order: initial; grid-column: auto; } `; const StyledIconWarning = styled(IconWarning)` display: block; `; export default withProjects(SummaryContent);