123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219 |
- import type {CSSProperties} from 'react';
- import {Link} from 'react-router';
- import styled from '@emotion/styled';
- import {LinkButton} from 'sentry/components/button';
- import type {GridColumnHeader} from 'sentry/components/gridEditable';
- import GridEditable, {COL_WIDTH_UNDEFINED} from 'sentry/components/gridEditable';
- import {Tooltip} from 'sentry/components/tooltip';
- import {IconProfiling} from 'sentry/icons/iconProfiling';
- import {t} from 'sentry/locale';
- import EventView from 'sentry/utils/discover/eventView';
- import {
- generateEventSlug,
- generateLinkToEventInTraceView,
- } from 'sentry/utils/discover/urls';
- import {useLocation} from 'sentry/utils/useLocation';
- import useOrganization from 'sentry/utils/useOrganization';
- import {normalizeUrl} from 'sentry/utils/withDomainRequired';
- import {DurationComparisonCell} from 'sentry/views/starfish/components/samplesTable/common';
- import {DurationCell} from 'sentry/views/starfish/components/tableCells/durationCell';
- import ResourceSizeCell from 'sentry/views/starfish/components/tableCells/resourceSizeCell';
- import {
- OverflowEllipsisTextContainer,
- TextAlignRight,
- } from 'sentry/views/starfish/components/textAlign';
- import type {SpanSample} from 'sentry/views/starfish/queries/useSpanSamples';
- import {SpanMetricsField} from 'sentry/views/starfish/types';
- const {HTTP_RESPONSE_CONTENT_LENGTH} = SpanMetricsField;
- type Keys =
- | 'transaction_id'
- | 'span_id'
- | 'profile_id'
- | 'timestamp'
- | 'duration'
- | 'p95_comparison'
- | 'avg_comparison'
- | 'http.response_content_length';
- export type SamplesTableColumnHeader = GridColumnHeader<Keys>;
- export const DEFAULT_COLUMN_ORDER: SamplesTableColumnHeader[] = [
- {
- key: 'span_id',
- name: 'Span ID',
- width: COL_WIDTH_UNDEFINED,
- },
- {
- key: 'duration',
- name: 'Span Duration',
- width: COL_WIDTH_UNDEFINED,
- },
- {
- key: 'avg_comparison',
- name: 'Compared to Average',
- width: COL_WIDTH_UNDEFINED,
- },
- ];
- type SpanTableRow = {
- op: string;
- transaction: {
- id: string;
- 'project.name': string;
- timestamp: string;
- trace: string;
- 'transaction.duration': number;
- };
- } & SpanSample;
- type Props = {
- avg: number;
- data: SpanTableRow[];
- isLoading: boolean;
- columnOrder?: SamplesTableColumnHeader[];
- highlightedSpanId?: string;
- onMouseLeaveSample?: () => void;
- onMouseOverSample?: (sample: SpanSample) => void;
- };
- export function SpanSamplesTable({
- isLoading,
- data,
- avg,
- highlightedSpanId,
- onMouseLeaveSample,
- onMouseOverSample,
- columnOrder,
- }: Props) {
- const location = useLocation();
- const organization = useOrganization();
- function handleMouseOverBodyCell(row: SpanTableRow) {
- if (onMouseOverSample) {
- onMouseOverSample(row);
- }
- }
- function handleMouseLeave() {
- if (onMouseLeaveSample) {
- onMouseLeaveSample();
- }
- }
- function renderHeadCell(column: GridColumnHeader): React.ReactNode {
- if (
- column.key === 'p95_comparison' ||
- column.key === 'avg_comparison' ||
- column.key === 'duration'
- ) {
- return (
- <TextAlignRight>
- <OverflowEllipsisTextContainer>{column.name}</OverflowEllipsisTextContainer>
- </TextAlignRight>
- );
- }
- return <OverflowEllipsisTextContainer>{column.name}</OverflowEllipsisTextContainer>;
- }
- function renderBodyCell(column: GridColumnHeader, row: SpanTableRow): React.ReactNode {
- const shouldHighlight = row.span_id === highlightedSpanId;
- const commonProps = {
- style: (shouldHighlight ? {fontWeight: 'bold'} : {}) satisfies CSSProperties,
- onMouseEnter: () => handleMouseOverBodyCell(row),
- };
- if (column.key === 'span_id') {
- return (
- <Link
- to={generateLinkToEventInTraceView({
- eventSlug: generateEventSlug({
- id: row['transaction.id'],
- project: row.project,
- }),
- organization,
- location,
- eventView: EventView.fromLocation(location),
- dataRow: {
- id: row['transaction.id'],
- trace: row.transaction?.trace,
- timestamp: row.timestamp,
- },
- spanId: row.span_id,
- })}
- {...commonProps}
- >
- {row.span_id}
- </Link>
- );
- }
- if (column.key === HTTP_RESPONSE_CONTENT_LENGTH) {
- const size = parseInt(row[HTTP_RESPONSE_CONTENT_LENGTH], 10);
- return <ResourceSizeCell bytes={size} />;
- }
- if (column.key === 'profile_id') {
- return (
- <IconWrapper>
- {row.profile_id ? (
- <Tooltip title={t('View Profile')}>
- <LinkButton
- to={normalizeUrl(
- `/organizations/${organization.slug}/profiling/profile/${row.project}/${row.profile_id}/flamegraph/?spanId=${row.span_id}`
- )}
- size="xs"
- >
- <IconProfiling size="xs" />
- </LinkButton>
- </Tooltip>
- ) : (
- <div {...commonProps}>(no value)</div>
- )}
- </IconWrapper>
- );
- }
- if (column.key === 'duration') {
- return (
- <DurationCell containerProps={commonProps} milliseconds={row['span.self_time']} />
- );
- }
- if (column.key === 'avg_comparison') {
- return (
- <DurationComparisonCell
- containerProps={commonProps}
- duration={row['span.self_time']}
- compareToDuration={avg}
- />
- );
- }
- return <span {...commonProps}>{row[column.key]}</span>;
- }
- return (
- <div onMouseLeave={handleMouseLeave}>
- <GridEditable
- isLoading={isLoading}
- data={data}
- columnOrder={columnOrder ?? DEFAULT_COLUMN_ORDER}
- columnSortBy={[]}
- grid={{
- renderHeadCell,
- renderBodyCell,
- }}
- location={location}
- />
- </div>
- );
- }
- const IconWrapper = styled('div')`
- text-align: right;
- width: 100%;
- height: 26px;
- `;
|