spanSamplesTable.tsx 3.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141
  1. import type {ComponentProps} from 'react';
  2. import styled from '@emotion/styled';
  3. import type {Location} from 'history';
  4. import GridEditable, {
  5. COL_WIDTH_UNDEFINED,
  6. type GridColumnHeader,
  7. } from 'sentry/components/gridEditable';
  8. import {t} from 'sentry/locale';
  9. import type {Organization} from 'sentry/types/organization';
  10. import type {EventsMetaType} from 'sentry/utils/discover/eventView';
  11. import {getFieldRenderer} from 'sentry/utils/discover/fieldRenderers';
  12. import {useLocation} from 'sentry/utils/useLocation';
  13. import useOrganization from 'sentry/utils/useOrganization';
  14. import {renderHeadCell} from 'sentry/views/starfish/components/tableCells/renderHeadCell';
  15. import {SpanIdCell} from 'sentry/views/starfish/components/tableCells/spanIdCell';
  16. import type {IndexedResponse} from 'sentry/views/starfish/types';
  17. import {SpanIndexedField} from 'sentry/views/starfish/types';
  18. type DataRowKeys =
  19. | SpanIndexedField.PROJECT
  20. | SpanIndexedField.TRANSACTION_ID
  21. | SpanIndexedField.TRACE
  22. | SpanIndexedField.TIMESTAMP
  23. | SpanIndexedField.ID
  24. | SpanIndexedField.SPAN_DESCRIPTION
  25. | SpanIndexedField.RESPONSE_CODE;
  26. type ColumnKeys =
  27. | SpanIndexedField.ID
  28. | SpanIndexedField.SPAN_DESCRIPTION
  29. | SpanIndexedField.RESPONSE_CODE;
  30. type DataRow = Pick<IndexedResponse, DataRowKeys>;
  31. type Column = GridColumnHeader<ColumnKeys>;
  32. const COLUMN_ORDER: Column[] = [
  33. {
  34. key: SpanIndexedField.ID,
  35. name: t('Span ID'),
  36. width: 150,
  37. },
  38. {
  39. key: SpanIndexedField.RESPONSE_CODE,
  40. name: t('Status'),
  41. width: 50,
  42. },
  43. {
  44. key: SpanIndexedField.SPAN_DESCRIPTION,
  45. name: t('URL'),
  46. width: COL_WIDTH_UNDEFINED,
  47. },
  48. ];
  49. interface Props {
  50. data: DataRow[];
  51. isLoading: boolean;
  52. error?: Error | null;
  53. highlightedSpanId?: string;
  54. meta?: EventsMetaType;
  55. onSampleMouseOut?: ComponentProps<typeof GridEditable>['onRowMouseOut'];
  56. onSampleMouseOver?: ComponentProps<typeof GridEditable>['onRowMouseOver'];
  57. }
  58. export function SpanSamplesTable({
  59. data,
  60. isLoading,
  61. error,
  62. meta,
  63. onSampleMouseOver,
  64. onSampleMouseOut,
  65. highlightedSpanId,
  66. }: Props) {
  67. const location = useLocation();
  68. const organization = useOrganization();
  69. return (
  70. <GridEditable
  71. aria-label={t('Span Samples')}
  72. isLoading={isLoading}
  73. error={error}
  74. data={data}
  75. columnOrder={COLUMN_ORDER}
  76. columnSortBy={[]}
  77. grid={{
  78. renderHeadCell: col =>
  79. renderHeadCell({
  80. column: col,
  81. location,
  82. }),
  83. renderBodyCell: (column, row) =>
  84. renderBodyCell(column, row, meta, location, organization),
  85. }}
  86. highlightedRowKey={data.findIndex(row => row.span_id === highlightedSpanId)}
  87. onRowMouseOver={onSampleMouseOver}
  88. onRowMouseOut={onSampleMouseOut}
  89. location={location}
  90. />
  91. );
  92. }
  93. function renderBodyCell(
  94. column: Column,
  95. row: DataRow,
  96. meta: EventsMetaType | undefined,
  97. location: Location,
  98. organization: Organization
  99. ) {
  100. if (column.key === SpanIndexedField.ID) {
  101. return (
  102. <SpanIdCell
  103. projectSlug={row.project}
  104. traceId={row.trace}
  105. timestamp={row.timestamp}
  106. transactionId={row[SpanIndexedField.TRANSACTION_ID]}
  107. spanId={row[SpanIndexedField.ID]}
  108. />
  109. );
  110. }
  111. if (column.key === SpanIndexedField.SPAN_DESCRIPTION) {
  112. return <SpanDescriptionCell>{row[column.key]}</SpanDescriptionCell>;
  113. }
  114. if (!meta?.fields) {
  115. return row[column.key];
  116. }
  117. const renderer = getFieldRenderer(column.key, meta.fields, false);
  118. return renderer(row, {
  119. location,
  120. organization,
  121. unit: meta.units?.[column.key],
  122. });
  123. }
  124. const SpanDescriptionCell = styled('span')`
  125. word-break: break-word;
  126. `;