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 (
{label}
);
}
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: (
),
}
);
return (
{t('TTID, TTFD')}
);
}
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: (
),
}
);
return (
{t('TTFD')}
);
}
const tooltipValue = tct(
'This span may not be contributing to TTID or TTFD. [link: Learn more.]',
{
link: (
),
}
);
return (
{'--'}
);
}
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: (
),
}
);
return (
{t('Yes')}
);
}
const tooltipValue = tct(
'This span may not affect initial display. [link: Learn more.]',
{
link: (
),
}
);
return (
{t('No')}
);
}
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 = (
);
return sortLink;
}
const columnSortBy = eventView.getSorts();
const handleCursor: CursorHandler = (newCursor, pathname, query) => {
browserHistory.push({
pathname,
query: {...query, [MobileCursors.SPANS_TABLE]: newCursor},
});
};
return (
{
return {key: col, name: columnNameMap[col] ?? col, width: COL_WIDTH_UNDEFINED};
})}
columnSortBy={columnSortBy}
grid={{
renderHeadCell: column => renderHeadCell(column, data?.meta),
renderBodyCell,
}}
/>
);
}
const Container = styled('div')`
${p => p.theme.overflowEllipsis};
text-align: right;
`;