transactionSamplesTable.tsx 7.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272
  1. import type {CSSProperties} from 'react';
  2. import {Fragment} from 'react';
  3. import styled from '@emotion/styled';
  4. import {DateTime} from 'sentry/components/dateTime';
  5. import type {GridColumnHeader} from 'sentry/components/gridEditable';
  6. import GridEditable, {COL_WIDTH_UNDEFINED} from 'sentry/components/gridEditable';
  7. import Link from 'sentry/components/links/link';
  8. import QuestionTooltip from 'sentry/components/questionTooltip';
  9. import {t} from 'sentry/locale';
  10. import type {NewQuery} from 'sentry/types';
  11. import EventView from 'sentry/utils/discover/eventView';
  12. import {getFieldRenderer} from 'sentry/utils/discover/fieldRenderers';
  13. import {SPAN_OP_RELATIVE_BREAKDOWN_FIELD} from 'sentry/utils/discover/fields';
  14. import {DiscoverDatasets} from 'sentry/utils/discover/types';
  15. import {
  16. generateEventSlug,
  17. generateLinkToEventInTraceView,
  18. } from 'sentry/utils/discover/urls';
  19. import {MutableSearch} from 'sentry/utils/tokenizeSearch';
  20. import {useLocation} from 'sentry/utils/useLocation';
  21. import useOrganization from 'sentry/utils/useOrganization';
  22. import {normalizeUrl} from 'sentry/utils/withDomainRequired';
  23. import {DurationComparisonCell} from 'sentry/views/starfish/components/samplesTable/common';
  24. import useErrorSamples from 'sentry/views/starfish/components/samplesTable/useErrorSamples';
  25. import {DurationCell} from 'sentry/views/starfish/components/tableCells/durationCell';
  26. import {
  27. OverflowEllipsisTextContainer,
  28. TextAlignLeft,
  29. TextAlignRight,
  30. } from 'sentry/views/starfish/components/textAlign';
  31. import {DataTitles} from 'sentry/views/starfish/views/spans/types';
  32. import type {SampleFilter} from 'sentry/views/starfish/views/webServiceView/endpointOverview';
  33. type Keys =
  34. | 'id'
  35. | 'profile_id'
  36. | 'timestamp'
  37. | 'transaction.duration'
  38. | 'avg_comparison'
  39. | 'span_ops_breakdown.relative'
  40. | 'http.status_code'
  41. | 'transaction.status';
  42. type TableColumnHeader = GridColumnHeader<Keys>;
  43. type Props = {
  44. data: DataRow[];
  45. isLoading: boolean;
  46. queryConditions: string[];
  47. sampleFilter: SampleFilter;
  48. setHighlightedId: (string) => void;
  49. averageDuration?: number;
  50. highlightedId?: string;
  51. };
  52. export type DataRow = {
  53. 'http.status_code': number;
  54. id: string;
  55. profile_id: string;
  56. 'spans.browser': number;
  57. 'spans.db': number;
  58. 'spans.http': number;
  59. 'spans.resource': number;
  60. 'spans.ui': number;
  61. timestamp: string;
  62. 'transaction.duration': number;
  63. 'transaction.status': string;
  64. };
  65. export function TransactionSamplesTable({
  66. queryConditions,
  67. sampleFilter,
  68. data,
  69. isLoading,
  70. averageDuration,
  71. setHighlightedId,
  72. highlightedId,
  73. }: Props) {
  74. const location = useLocation();
  75. const organization = useOrganization();
  76. const query = new MutableSearch(queryConditions);
  77. const savedQuery: NewQuery = {
  78. id: undefined,
  79. name: 'Endpoint Overview Samples',
  80. query: query.formatString(),
  81. fields: [],
  82. dataset: DiscoverDatasets.DISCOVER,
  83. version: 2,
  84. };
  85. const columnOrder: TableColumnHeader[] = [
  86. {
  87. key: 'id',
  88. name: 'Event ID',
  89. width: 100,
  90. },
  91. ...(sampleFilter === 'ALL'
  92. ? [
  93. {
  94. key: 'profile_id',
  95. name: 'Profile ID',
  96. width: 140,
  97. } as TableColumnHeader,
  98. {
  99. key: SPAN_OP_RELATIVE_BREAKDOWN_FIELD,
  100. name: 'Operation Duration',
  101. width: 180,
  102. } as TableColumnHeader,
  103. {
  104. key: 'timestamp',
  105. name: 'Timestamp',
  106. width: 230,
  107. } as TableColumnHeader,
  108. {
  109. key: 'transaction.duration',
  110. name: DataTitles.duration,
  111. width: 100,
  112. } as TableColumnHeader,
  113. {
  114. key: 'avg_comparison',
  115. name: 'Compared to Average',
  116. width: 100,
  117. } as TableColumnHeader,
  118. ]
  119. : [
  120. {
  121. key: 'http.status_code',
  122. name: 'Response Code',
  123. width: COL_WIDTH_UNDEFINED,
  124. } as TableColumnHeader,
  125. {
  126. key: 'transaction.status',
  127. name: 'Status',
  128. width: COL_WIDTH_UNDEFINED,
  129. } as TableColumnHeader,
  130. {
  131. key: 'timestamp',
  132. name: 'Timestamp',
  133. width: 230,
  134. } as TableColumnHeader,
  135. ]),
  136. ];
  137. const eventView = EventView.fromNewQueryWithLocation(savedQuery, location);
  138. const {isLoading: isErrorSamplesLoading, data: errorSamples} =
  139. useErrorSamples(eventView);
  140. function renderHeadCell(column: GridColumnHeader): React.ReactNode {
  141. if (column.key === 'avg_comparison' || column.key === 'transaction.duration') {
  142. return (
  143. <TextAlignRight>
  144. <OverflowEllipsisTextContainer>{column.name}</OverflowEllipsisTextContainer>
  145. </TextAlignRight>
  146. );
  147. }
  148. if (column.key === SPAN_OP_RELATIVE_BREAKDOWN_FIELD) {
  149. return (
  150. <Fragment>
  151. {column.name}
  152. <StyledIconQuestion
  153. size="xs"
  154. position="top"
  155. title={t(
  156. `Span durations are summed over the course of an entire transaction. Any overlapping spans are only counted once.`
  157. )}
  158. />
  159. </Fragment>
  160. );
  161. }
  162. return <OverflowEllipsisTextContainer>{column.name}</OverflowEllipsisTextContainer>;
  163. }
  164. function renderBodyCell(column: TableColumnHeader, row: DataRow): React.ReactNode {
  165. const commonProps = {
  166. style: (row.id === highlightedId
  167. ? {fontWeight: 'bold'}
  168. : {}) satisfies CSSProperties,
  169. onMouseEnter: () => setHighlightedId(row.id),
  170. };
  171. if (column.key === 'id') {
  172. return (
  173. <Link
  174. {...commonProps}
  175. to={generateLinkToEventInTraceView({
  176. eventSlug: generateEventSlug(row),
  177. organization,
  178. location,
  179. eventView,
  180. dataRow: row,
  181. })}
  182. >
  183. {row.id.slice(0, 8)}
  184. </Link>
  185. );
  186. }
  187. if (column.key === 'profile_id') {
  188. return row.profile_id ? (
  189. <Link
  190. {...commonProps}
  191. to={normalizeUrl(
  192. `/organizations/${organization.slug}/profiling/profile/${row['project.name']}/${row.profile_id}/flamegraph/`
  193. )}
  194. >
  195. {row.profile_id.slice(0, 8)}
  196. </Link>
  197. ) : (
  198. <div {...commonProps}>(no value)</div>
  199. );
  200. }
  201. if (column.key === 'transaction.duration') {
  202. return <DurationCell milliseconds={row['transaction.duration']} />;
  203. }
  204. if (column.key === 'timestamp') {
  205. return <DateTime {...commonProps} date={row[column.key]} year timeZone seconds />;
  206. }
  207. if (column.key === 'avg_comparison') {
  208. return (
  209. <DurationComparisonContainer {...commonProps}>
  210. <DurationComparisonCell
  211. duration={row['transaction.duration']}
  212. compareToDuration={averageDuration ?? 0}
  213. />
  214. </DurationComparisonContainer>
  215. );
  216. }
  217. if (column.key === SPAN_OP_RELATIVE_BREAKDOWN_FIELD) {
  218. return getFieldRenderer(column.key, {})(row, {
  219. location,
  220. organization,
  221. eventView,
  222. });
  223. }
  224. return <TextAlignLeft {...commonProps}>{row[column.key]}</TextAlignLeft>;
  225. }
  226. return (
  227. <div onMouseLeave={() => setHighlightedId(undefined)}>
  228. <GridEditable
  229. isLoading={sampleFilter === 'ALL' ? isLoading : isErrorSamplesLoading}
  230. data={sampleFilter === 'ALL' ? (data as DataRow[]) : (errorSamples as DataRow[])}
  231. columnOrder={columnOrder}
  232. columnSortBy={[]}
  233. location={location}
  234. grid={{
  235. renderHeadCell,
  236. renderBodyCell,
  237. }}
  238. />
  239. </div>
  240. );
  241. }
  242. const DurationComparisonContainer = styled('div')`
  243. text-align: right;
  244. width: 100%;
  245. display: inline-block;
  246. `;
  247. const StyledIconQuestion = styled(QuestionTooltip)`
  248. position: relative;
  249. left: 4px;
  250. `;