import type {ReactNode} from 'react'; import styled from '@emotion/styled'; import type {Location} from 'history'; import Count from 'sentry/components/count'; import type {GridColumnOrder} from 'sentry/components/gridEditable'; import GridEditable, {COL_WIDTH_UNDEFINED} from 'sentry/components/gridEditable'; import SortLink from 'sentry/components/gridEditable/sortLink'; import {IconArrow} from 'sentry/icons'; import {t} from 'sentry/locale'; import {space} from 'sentry/styles/space'; import type {Organization} from 'sentry/types/organization'; import {getFieldRenderer} from 'sentry/utils/discover/fieldRenderers'; import type {ColumnType} from 'sentry/utils/discover/fields'; import {fieldAlignment} from 'sentry/utils/discover/fields'; import type {AnomalyInfo} from 'sentry/utils/performance/anomalies/anomaliesQuery'; type Props = { isLoading: boolean; location: Location; organization: Organization; anomalies?: AnomalyInfo[]; }; const transformRow = (anom: AnomalyInfo): TableDataRowWithExtras => { return { anomaly: `#${anom.id}`, confidence: anom.confidence, timestamp: new Date(anom.start), timeInterval: anom.end - anom.start, expected: anom.expected, received: anom.received, }; }; export default function AnomaliesTable(props: Props) { const {location, organization, isLoading, anomalies} = props; const data: TableDataRowWithExtras[] = anomalies?.map(transformRow) || []; return ( ); } function renderHeadCell(column: TableColumn, _index: number): ReactNode { const align = fieldAlignment(column.key, COLUMN_TYPE[column.key]); return ( undefined} /> ); } function renderBodyCellWithMeta(location: Location, organization: Organization) { return function ( column: TableColumn, dataRow: TableDataRowWithExtras ): React.ReactNode { const fieldRenderer = getFieldRenderer(column.key, COLUMN_TYPE); if (column.key === 'confidence') { return ( {dataRow.confidence === 'low' ? ( {t('Low Confidence')} ) : ( {t('High Confidence')} )} ); } if (column.key === 'expected') { return ( ); } if (column.key === 'received') { return ( dataRow.expected ? 'up' : 'down'} /> ); } return fieldRenderer(dataRow, {location, organization}); }; } const NumberCell = styled('div')` display: flex; justify-content: flex-end; align-items: center; gap: ${space(0.5)}; `; const LowConfidence = styled('div')` color: ${p => p.theme.yellow400}; `; const HighConfidence = styled('div')` color: ${p => p.theme.red400}; `; const ConfidenceCell = styled('div')` text-align: left; justify-self: flex-end; flex-grow: 1; `; type TableColumnKey = | 'anomaly' | 'confidence' | 'timeInterval' | 'timestamp' | 'expected' | 'received'; type TableColumn = GridColumnOrder; type TableDataRow = Record; type TableDataRowWithExtras = TableDataRow & {}; const COLUMNS: Record = { anomaly: { key: 'anomaly', name: t('Anomaly'), width: COL_WIDTH_UNDEFINED, }, confidence: { key: 'confidence', name: t('Confidence'), width: COL_WIDTH_UNDEFINED, }, timeInterval: { key: 'timeInterval', name: t('Time Interval'), width: COL_WIDTH_UNDEFINED, }, timestamp: { key: 'timestamp', name: t('Timestamp'), width: COL_WIDTH_UNDEFINED, }, expected: { key: 'expected', name: t('Expected'), width: COL_WIDTH_UNDEFINED, }, received: { key: 'received', name: t('Received'), width: COL_WIDTH_UNDEFINED, }, }; const COLUMN_TYPE: Record = { anomaly: 'string', confidence: 'string', timeInterval: 'duration', timestamp: 'date', expected: 'number', received: 'number', };