123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393 |
- import {Fragment, useMemo} from 'react';
- import styled from '@emotion/styled';
- import * as qs from 'query-string';
- import {getInterval} from 'sentry/components/charts/utils';
- import type {GridColumnHeader} 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 type {CursorHandler} from 'sentry/components/pagination';
- import Pagination from 'sentry/components/pagination';
- import {Tooltip} from 'sentry/components/tooltip';
- import {t, tct} from 'sentry/locale';
- import type {NewQuery} from 'sentry/types/organization';
- import {browserHistory} from 'sentry/utils/browserHistory';
- import type {TableDataRow} from 'sentry/utils/discover/discoverQuery';
- import type {MetaType} from 'sentry/utils/discover/eventView';
- import EventView, {isFieldSortable} from 'sentry/utils/discover/eventView';
- import {getFieldRenderer} from 'sentry/utils/discover/fieldRenderers';
- import type {Sort} from 'sentry/utils/discover/fields';
- import {fieldAlignment} from 'sentry/utils/discover/fields';
- import {DiscoverDatasets} from 'sentry/utils/discover/types';
- import {decodeScalar, decodeSorts} from 'sentry/utils/queryString';
- import {MutableSearch} from 'sentry/utils/tokenizeSearch';
- import {useLocation} from 'sentry/utils/useLocation';
- import useOrganization from 'sentry/utils/useOrganization';
- import usePageFilters from 'sentry/utils/usePageFilters';
- import {
- PRIMARY_RELEASE_ALIAS,
- SECONDARY_RELEASE_ALIAS,
- } from 'sentry/views/insights/common/components/releaseSelector';
- import {OverflowEllipsisTextContainer} from 'sentry/views/insights/common/components/textAlign';
- import {useTTFDConfigured} from 'sentry/views/insights/common/queries/useHasTtfdConfigured';
- import {STARFISH_CHART_INTERVAL_FIDELITY} from 'sentry/views/insights/common/utils/constants';
- import {appendReleaseFilters} from 'sentry/views/insights/common/utils/releaseComparison';
- import {useModuleURL} from 'sentry/views/insights/common/utils/useModuleURL';
- import {QueryParameterNames} from 'sentry/views/insights/common/views/queryParameters';
- import useCrossPlatformProject from 'sentry/views/insights/mobile/common/queries/useCrossPlatformProject';
- import {
- SpanOpSelector,
- TTID_CONTRIBUTING_SPAN_OPS,
- } from 'sentry/views/insights/mobile/screenload/components/spanOpSelector';
- import {useTableQuery} from 'sentry/views/insights/mobile/screenload/components/tables/screensTable';
- import {MobileCursors} from 'sentry/views/insights/mobile/screenload/constants';
- import {MODULE_DOC_LINK} from 'sentry/views/insights/mobile/screenload/settings';
- import {SpanMetricsField} from 'sentry/views/insights/types';
- const {SPAN_SELF_TIME, SPAN_DESCRIPTION, SPAN_GROUP, SPAN_OP, PROJECT_ID} =
- SpanMetricsField;
- type Props = {
- primaryRelease?: string;
- secondaryRelease?: string;
- transaction?: string;
- };
- export function ScreenLoadSpansTable({
- transaction,
- primaryRelease,
- secondaryRelease,
- }: Props) {
- const moduleURL = useModuleURL('screen_load');
- const location = useLocation();
- const {selection} = usePageFilters();
- const organization = useOrganization();
- const cursor = decodeScalar(location.query?.[MobileCursors.SPANS_TABLE]);
- const {isProjectCrossPlatform, selectedPlatform} = useCrossPlatformProject();
- const spanOp = decodeScalar(location.query[SpanMetricsField.SPAN_OP]) ?? '';
- const {hasTTFD, isLoading: hasTTFDLoading} = useTTFDConfigured([
- `transaction:"${transaction}"`,
- ]);
- const queryStringPrimary = useMemo(() => {
- const searchQuery = new MutableSearch([
- 'transaction.op:ui.load',
- `transaction:${transaction}`,
- 'has:span.description',
- ...(spanOp
- ? [`${SpanMetricsField.SPAN_OP}:${spanOp}`]
- : [`span.op:[${TTID_CONTRIBUTING_SPAN_OPS.join(',')}]`]),
- ]);
- if (isProjectCrossPlatform) {
- searchQuery.addFilterValue('os.name', selectedPlatform);
- }
- return appendReleaseFilters(searchQuery, primaryRelease, secondaryRelease);
- }, [
- isProjectCrossPlatform,
- primaryRelease,
- secondaryRelease,
- selectedPlatform,
- spanOp,
- transaction,
- ]);
- const sort = decodeSorts(location.query[QueryParameterNames.SPANS_SORT])[0] ?? {
- kind: 'desc',
- field: 'time_spent_percentage()',
- };
- const newQuery: NewQuery = {
- name: '',
- fields: [
- PROJECT_ID,
- SPAN_OP,
- SPAN_GROUP,
- SPAN_DESCRIPTION,
- `avg_if(${SPAN_SELF_TIME},release,${primaryRelease})`,
- `avg_if(${SPAN_SELF_TIME},release,${secondaryRelease})`,
- 'ttid_contribution_rate()',
- 'ttfd_contribution_rate()',
- 'count()',
- 'time_spent_percentage()',
- `sum(${SPAN_SELF_TIME})`,
- ],
- query: queryStringPrimary,
- dataset: DiscoverDatasets.SPANS_METRICS,
- version: 2,
- projects: selection.projects,
- interval: getInterval(selection.datetime, STARFISH_CHART_INTERVAL_FIDELITY),
- };
- const eventView = EventView.fromNewQueryWithLocation(newQuery, location);
- eventView.sorts = [sort];
- const {data, isLoading, pageLinks} = useTableQuery({
- eventView,
- enabled: true,
- referrer: 'api.starfish.mobile-span-table',
- cursor,
- });
- const columnNameMap = {
- [SPAN_OP]: t('Operation'),
- [SPAN_DESCRIPTION]: t('Span Description'),
- 'count()': t('Total Count'),
- affects: hasTTFD ? t('Affects') : t('Affects TTID'),
- 'time_spent_percentage()': t('Total Time Spent'),
- [`avg_if(${SPAN_SELF_TIME},release,${primaryRelease})`]: t(
- 'Avg Duration (%s)',
- PRIMARY_RELEASE_ALIAS
- ),
- [`avg_if(${SPAN_SELF_TIME},release,${secondaryRelease})`]: t(
- 'Avg Duration (%s)',
- SECONDARY_RELEASE_ALIAS
- ),
- };
- function renderBodyCell(column, row): React.ReactNode {
- if (!data?.meta || !data?.meta.fields) {
- return row[column.key];
- }
- if (column.key === SPAN_DESCRIPTION) {
- const label = row[SpanMetricsField.SPAN_DESCRIPTION];
- const pathname = `${moduleURL}/spans/`;
- const query = {
- ...location.query,
- transaction,
- spanGroup: row[SpanMetricsField.SPAN_GROUP],
- spanDescription: row[SpanMetricsField.SPAN_DESCRIPTION],
- };
- return (
- <OverflowEllipsisTextContainer>
- <Link to={`${pathname}?${qs.stringify(query)}`}>{label}</Link>
- </OverflowEllipsisTextContainer>
- );
- }
- if (column.key === 'affects' && hasTTFD) {
- const ttid_contribution_rate = row['ttid_contribution_rate()']
- ? parseFloat(row['ttid_contribution_rate()'])
- : 0;
- const ttfd_contribution_rate = row['ttfd_contribution_rate()']
- ? parseFloat(row['ttfd_contribution_rate()'])
- : 0;
- if (!isNaN(ttid_contribution_rate) && ttid_contribution_rate > 0.99) {
- const tooltipValue = tct(
- 'This span always ends before TTID and TTFD and may affect initial and final display. [link: Learn more.]',
- {
- link: (
- <ExternalLink href={`${MODULE_DOC_LINK}#ttid-and-ttfd-affecting-spans`} />
- ),
- }
- );
- return (
- <Tooltip isHoverable title={tooltipValue} showUnderline>
- <Container>{t('TTID, TTFD')}</Container>
- </Tooltip>
- );
- }
- if (!isNaN(ttfd_contribution_rate) && ttfd_contribution_rate > 0.99) {
- const tooltipValue = tct(
- 'This span always ends before TTFD and may affect final display. [link: Learn more.]',
- {
- link: (
- <ExternalLink href={`${MODULE_DOC_LINK}#ttid-and-ttfd-affecting-spans`} />
- ),
- }
- );
- return (
- <Tooltip isHoverable title={tooltipValue} showUnderline>
- <Container>{t('TTFD')}</Container>
- </Tooltip>
- );
- }
- const tooltipValue = tct(
- 'This span may not be contributing to TTID or TTFD. [link: Learn more.]',
- {
- link: (
- <ExternalLink href={`${MODULE_DOC_LINK}#ttid-and-ttfd-affecting-spans`} />
- ),
- }
- );
- return (
- <Tooltip isHoverable title={tooltipValue}>
- <Container>{'--'}</Container>
- </Tooltip>
- );
- }
- if (column.key === 'affects') {
- const ttid_contribution_rate = row['ttid_contribution_rate()']
- ? parseFloat(row['ttid_contribution_rate()'])
- : 0;
- if (!isNaN(ttid_contribution_rate) && ttid_contribution_rate > 0.99) {
- const tooltipValue = tct(
- 'This span always ends before TTID and may affect initial display. [link: Learn more.]',
- {
- link: (
- <ExternalLink href={`${MODULE_DOC_LINK}#ttid-and-ttfd-affecting-spans`} />
- ),
- }
- );
- return (
- <Tooltip isHoverable title={tooltipValue} showUnderline>
- <Container>{t('Yes')}</Container>
- </Tooltip>
- );
- }
- const tooltipValue = tct(
- 'This span may not affect initial display. [link: Learn more.]',
- {
- link: (
- <ExternalLink href={`${MODULE_DOC_LINK}#ttid-and-ttfd-affecting-spans`} />
- ),
- }
- );
- return (
- <Tooltip isHoverable title={tooltipValue} showUnderline>
- <Container>{t('No')}</Container>
- </Tooltip>
- );
- }
- const renderer = getFieldRenderer(column.key, data?.meta.fields, false);
- const rendered = renderer(row, {
- location,
- organization,
- unit: data?.meta.units?.[column.key],
- });
- return rendered;
- }
- function renderHeadCell(
- column: GridColumnHeader,
- tableMeta?: MetaType
- ): React.ReactNode {
- const fieldType = tableMeta?.fields?.[column.key];
- let alignment = fieldAlignment(column.key as string, fieldType);
- if (column.key === 'affects') {
- alignment = 'right';
- }
- const field = {
- field: column.key as string,
- width: column.width,
- };
- const affectsIsCurrentSort =
- column.key === 'affects' &&
- (sort?.field === 'ttid_contribution_rate()' ||
- sort?.field === 'ttfd_contribution_rate()');
- function generateSortLink() {
- if (!tableMeta) {
- return undefined;
- }
- let newSortDirection: Sort['kind'] = 'desc';
- if (sort?.field === column.key) {
- if (sort.kind === 'desc') {
- newSortDirection = 'asc';
- }
- }
- function getNewSort() {
- if (column.key === 'affects') {
- if (sort?.field === 'ttid_contribution_rate()') {
- return '-ttfd_contribution_rate()';
- }
- return '-ttid_contribution_rate()';
- }
- return `${newSortDirection === 'desc' ? '-' : ''}${column.key}`;
- }
- return {
- ...location,
- query: {...location.query, [QueryParameterNames.SPANS_SORT]: getNewSort()},
- };
- }
- const canSort =
- column.key === 'affects' || isFieldSortable(field, tableMeta?.fields, true);
- const sortLink = (
- <SortLink
- align={alignment}
- title={column.name}
- direction={
- affectsIsCurrentSort
- ? sort?.field === 'ttid_contribution_rate()'
- ? 'desc'
- : 'asc'
- : sort?.field === column.key
- ? sort.kind
- : undefined
- }
- canSort={canSort}
- generateSortLink={generateSortLink}
- />
- );
- return sortLink;
- }
- const columnSortBy = eventView.getSorts();
- const handleCursor: CursorHandler = (newCursor, pathname, query) => {
- browserHistory.push({
- pathname,
- query: {...query, [MobileCursors.SPANS_TABLE]: newCursor},
- });
- };
- return (
- <Fragment>
- <SpanOpSelector
- primaryRelease={primaryRelease}
- transaction={transaction}
- secondaryRelease={secondaryRelease}
- />
- <GridEditable
- isLoading={isLoading || hasTTFDLoading}
- data={data?.data as TableDataRow[]}
- columnOrder={[
- String(SPAN_OP),
- String(SPAN_DESCRIPTION),
- `avg_if(${SPAN_SELF_TIME},release,${primaryRelease})`,
- `avg_if(${SPAN_SELF_TIME},release,${secondaryRelease})`,
- ...(organization.features.includes('insights-initial-modules')
- ? ['affects']
- : []),
- ...['count()', 'time_spent_percentage()'],
- ].map(col => {
- return {key: col, name: columnNameMap[col] ?? col, width: COL_WIDTH_UNDEFINED};
- })}
- columnSortBy={columnSortBy}
- grid={{
- renderHeadCell: column => renderHeadCell(column, data?.meta),
- renderBodyCell,
- }}
- />
- <Pagination pageLinks={pageLinks} onCursor={handleCursor} />
- </Fragment>
- );
- }
- const Container = styled('div')`
- ${p => p.theme.overflowEllipsis};
- text-align: right;
- `;
|