import {useMemo} from 'react'; import styled from '@emotion/styled'; import ProjectAvatar from 'sentry/components/avatar/projectAvatar'; import {Button} from 'sentry/components/button'; import ButtonBar from 'sentry/components/buttonBar'; import type {GridColumnHeader, GridColumnOrder} from 'sentry/components/gridEditable'; import GridEditable, {COL_WIDTH_UNDEFINED} from 'sentry/components/gridEditable'; import SortLink from 'sentry/components/gridEditable/sortLink'; import ExternalLink from 'sentry/components/links/externalLink'; import Link from 'sentry/components/links/link'; import Pagination from 'sentry/components/pagination'; import SearchBar from 'sentry/components/searchBar'; import {Tooltip} from 'sentry/components/tooltip'; import {IconChevron} from 'sentry/icons/iconChevron'; import {t} from 'sentry/locale'; import {space} from 'sentry/styles/space'; import {browserHistory} from 'sentry/utils/browserHistory'; import type {Sort} from 'sentry/utils/discover/fields'; import {parseFunction} from 'sentry/utils/discover/fields'; import getDuration from 'sentry/utils/duration/getDuration'; import {formatAbbreviatedNumber} from 'sentry/utils/formatters'; import {decodeScalar} from 'sentry/utils/queryString'; import {escapeFilterValue} from 'sentry/utils/tokenizeSearch'; import {useLocation} from 'sentry/utils/useLocation'; import useProjects from 'sentry/utils/useProjects'; import {PerformanceBadge} from 'sentry/views/performance/browser/webVitals/components/performanceBadge'; import {MODULE_DOC_LINK} from 'sentry/views/performance/browser/webVitals/settings'; import {useProjectWebVitalsScoresQuery} from 'sentry/views/performance/browser/webVitals/utils/queries/storedScoreQueries/useProjectWebVitalsScoresQuery'; import {useTransactionWebVitalsQuery} from 'sentry/views/performance/browser/webVitals/utils/queries/useTransactionWebVitalsQuery'; import type {RowWithScoreAndOpportunity} from 'sentry/views/performance/browser/webVitals/utils/types'; import {SORTABLE_FIELDS} from 'sentry/views/performance/browser/webVitals/utils/types'; import {useWebVitalsSort} from 'sentry/views/performance/browser/webVitals/utils/useWebVitalsSort'; type Column = GridColumnHeader; const COLUMN_ORDER: GridColumnOrder[] = [ {key: 'transaction', width: COL_WIDTH_UNDEFINED, name: 'Pages'}, {key: 'count()', width: COL_WIDTH_UNDEFINED, name: 'Pageloads'}, {key: 'p75(measurements.lcp)', width: COL_WIDTH_UNDEFINED, name: 'LCP'}, {key: 'p75(measurements.fcp)', width: COL_WIDTH_UNDEFINED, name: 'FCP'}, { key: 'p75(measurements.inp)', width: COL_WIDTH_UNDEFINED, name: 'INP', }, {key: 'p75(measurements.cls)', width: COL_WIDTH_UNDEFINED, name: 'CLS'}, {key: 'p75(measurements.ttfb)', width: COL_WIDTH_UNDEFINED, name: 'TTFB'}, {key: 'totalScore', width: COL_WIDTH_UNDEFINED, name: 'Score'}, {key: 'opportunity', width: COL_WIDTH_UNDEFINED, name: 'Opportunity'}, ]; const MAX_ROWS = 25; const DEFAULT_SORT: Sort = { field: 'opportunity_score(measurements.score.total)', kind: 'desc', }; export function PagePerformanceTable() { const location = useLocation(); const {projects} = useProjects(); const columnOrder = COLUMN_ORDER; const query = decodeScalar(location.query.query, ''); const project = useMemo( () => projects.find(p => p.id === String(location.query.project)), [projects, location.query.project] ); let sort = useWebVitalsSort({defaultSort: DEFAULT_SORT}); // Need to map fid back to inp for rendering if (sort.field === 'p75(measurements.fid)') { sort = {...sort, field: 'p75(measurements.inp)'}; } const {data: projectScoresData, isLoading: isProjectScoresLoading} = useProjectWebVitalsScoresQuery(); const { data, pageLinks, isLoading: isTransactionWebVitalsQueryLoading, } = useTransactionWebVitalsQuery({ limit: MAX_ROWS, transaction: query !== '' ? `*${escapeFilterValue(query)}*` : undefined, defaultSort: DEFAULT_SORT, shouldEscapeFilters: false, }); const scoreCount = projectScoresData?.data?.[0]?.[ 'count_scores(measurements.score.total)' ] as number; const tableData: RowWithScoreAndOpportunity[] = data.map(row => ({ ...row, opportunity: (((row as RowWithScoreAndOpportunity).opportunity ?? 0) * 100) / scoreCount, })); const getFormattedDuration = (value: number) => { return getDuration(value, value < 1 ? 0 : 2, true); }; function renderHeadCell(col: Column) { function generateSortLink() { const key = col.key === 'totalScore' ? 'avg(measurements.score.total)' : col.key === 'opportunity' ? 'opportunity_score(measurements.score.total)' : col.key; let newSortDirection: Sort['kind'] = 'desc'; if (sort?.field === key) { if (sort.kind === 'desc') { newSortDirection = 'asc'; } } const newSort = `${newSortDirection === 'desc' ? '-' : ''}${key}`; return { ...location, query: {...location.query, sort: newSort}, }; } const sortableFields = SORTABLE_FIELDS; const canSort = (sortableFields as unknown as string[]).includes(col.key); if (canSort && !['totalScore', 'opportunity'].includes(col.key)) { return ( ); } if (col.key === 'totalScore') { return ( {t('The overall performance rating of this page.')}
{t('How is this calculated?')} } > {t('Perf Score')}} direction={sort?.field === col.key ? sort.kind : undefined} canSort={canSort} generateSortLink={generateSortLink} align={undefined} />
); } if (col.key === 'opportunity') { return ( {t( "A number rating how impactful a performance improvement on this page would be to your application's overall Performance Score." )}
{t('How is this calculated?')} } > {col.name}} direction={sort?.field === col.key ? sort.kind : undefined} canSort={canSort} generateSortLink={generateSortLink} />
); } return {col.name}; } function renderBodyCell(col: Column, row: RowWithScoreAndOpportunity) { const {key} = col; if (key === 'totalScore') { return ( ); } if (key === 'count()') { return {formatAbbreviatedNumber(row['count()'])}; } if (key === 'transaction') { return ( {project && ( )} {row.transaction} ); } if ( [ 'p75(measurements.fcp)', 'p75(measurements.lcp)', 'p75(measurements.ttfb)', 'p75(measurements.fid)', 'p75(measurements.inp)', ].includes(key) ) { const measurement = parseFunction(key)?.arguments?.[0]; const func = 'count_scores'; const args = [measurement?.replace('measurements.', 'measurements.score.')]; const countWebVitalKey = `${func}(${args.join(', ')})`; const countWebVital = row[countWebVitalKey]; if (measurement === undefined || countWebVital === 0) { return ( {' \u2014 '} ); } return {getFormattedDuration((row[key] as number) / 1000)}; } if (key === 'p75(measurements.cls)') { const countWebVitalKey = 'count_scores(measurements.score.cls)'; const countWebVital = row[countWebVitalKey]; if (countWebVital === 0) { return ( {' \u2014 '} ); } return {Math.round((row[key] as number) * 100) / 100}; } if (key === 'opportunity') { if (row.opportunity !== undefined) { return ( {Math.round((row.opportunity as number) * 100) / 100} ); } return null; } return {row[key]}; } const handleSearch = (newQuery: string) => { browserHistory.push({ ...location, query: { ...location.query, query: newQuery === '' ? undefined : newQuery, cursor: undefined, }, }); }; return ( {/* The Pagination component disappears if pageLinks is not defined, which happens any time the table data is loading. So we render a disabled button bar if pageLinks is not defined to minimize ui shifting */} {!pageLinks && (