123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176 |
- import {Fragment, useEffect, useMemo} from 'react';
- import {useTheme} from '@emotion/react';
- import type {Location} from 'history';
- import * as Layout from 'sentry/components/layouts/thirds';
- import LoadingIndicator from 'sentry/components/loadingIndicator';
- import {t} from 'sentry/locale';
- import type {Organization} from 'sentry/types';
- import EventView from 'sentry/utils/discover/eventView';
- import {
- SPAN_OP_BREAKDOWN_FIELDS,
- SPAN_OP_RELATIVE_BREAKDOWN_FIELD,
- } from 'sentry/utils/discover/fields';
- import useReplayList from 'sentry/utils/replays/hooks/useReplayList';
- import {useLocation} from 'sentry/utils/useLocation';
- import useMedia from 'sentry/utils/useMedia';
- import useOrganization from 'sentry/utils/useOrganization';
- import useProjects from 'sentry/utils/useProjects';
- import type {ChildProps} from 'sentry/views/performance/transactionSummary/pageLayout';
- import PageLayout from 'sentry/views/performance/transactionSummary/pageLayout';
- import Tab from 'sentry/views/performance/transactionSummary/tabs';
- import ReplayTable from 'sentry/views/replays/replayTable';
- import {ReplayColumn} from 'sentry/views/replays/replayTable/types';
- import type {ReplayListLocationQuery} from 'sentry/views/replays/types';
- import type {EventSpanData} from './useReplaysFromTransaction';
- import useReplaysFromTransaction from './useReplaysFromTransaction';
- import useReplaysWithTxData from './useReplaysWithTxData';
- function TransactionReplays() {
- const location = useLocation<ReplayListLocationQuery>();
- const organization = useOrganization();
- const {projects} = useProjects();
- return (
- <PageLayout
- location={{
- ...location,
- query: {
- ...location.query,
- statsPeriod: '14d',
- },
- }}
- organization={organization}
- projects={projects}
- tab={Tab.REPLAYS}
- getDocumentTitle={getDocumentTitle}
- generateEventView={generateEventView}
- childComponent={ReplaysContentWrapper}
- />
- );
- }
- function getDocumentTitle(transactionName: string): string {
- const hasTransactionName =
- typeof transactionName === 'string' && String(transactionName).trim().length > 0;
- if (hasTransactionName) {
- return [String(transactionName).trim(), t('Replays')].join(' \u2014 ');
- }
- return [t('Summary'), t('Replays')].join(' \u2014 ');
- }
- function generateEventView({
- location,
- transactionName,
- }: {
- location: Location;
- transactionName: string;
- }) {
- const fields = [
- 'replayId',
- 'count()',
- 'transaction.duration',
- 'trace',
- 'timestamp',
- ...SPAN_OP_BREAKDOWN_FIELDS,
- SPAN_OP_RELATIVE_BREAKDOWN_FIELD,
- ];
- return EventView.fromSavedQuery({
- id: '',
- name: `Replay events within a transaction`,
- version: 2,
- fields,
- query: `event.type:transaction transaction:"${transactionName}" !replayId:""`,
- projects: [Number(location.query.project)],
- });
- }
- function ReplaysContentWrapper({
- eventView: replayIdsEventView,
- location,
- organization,
- setError,
- }: ChildProps) {
- const {data, fetchError, isFetching, pageLinks} = useReplaysFromTransaction({
- replayIdsEventView,
- location,
- organization,
- });
- useEffect(() => {
- setError(fetchError?.message ?? fetchError);
- }, [setError, fetchError]);
- if (!data) {
- return isFetching ? (
- <Layout.Main fullWidth>
- <LoadingIndicator />
- </Layout.Main>
- ) : (
- <Fragment>{null}</Fragment>
- );
- }
- const {events, replayRecordsEventView} = data;
- return (
- <ReplaysContent
- eventView={replayRecordsEventView}
- events={events}
- organization={organization}
- pageLinks={pageLinks}
- />
- );
- }
- function ReplaysContent({
- eventView,
- events,
- organization,
- }: {
- eventView: EventView;
- events: EventSpanData[];
- organization: Organization;
- pageLinks: string | null;
- }) {
- const location = useMemo(() => ({query: {}}) as Location<ReplayListLocationQuery>, []);
- const theme = useTheme();
- const hasRoomForColumns = useMedia(`(min-width: ${theme.breakpoints.small})`);
- const {replays, isFetching, fetchError} = useReplayList({
- eventView,
- location,
- organization,
- queryReferrer: 'transactionReplays',
- });
- const replaysWithTx = useReplaysWithTxData({
- replays,
- events,
- });
- return (
- <Layout.Main fullWidth>
- <ReplayTable
- fetchError={fetchError}
- isFetching={isFetching}
- replays={replaysWithTx}
- sort={undefined}
- visibleColumns={[
- ReplayColumn.REPLAY,
- ...(hasRoomForColumns ? [ReplayColumn.SLOWEST_TRANSACTION] : []),
- ReplayColumn.OS,
- ReplayColumn.BROWSER,
- ReplayColumn.DURATION,
- ReplayColumn.COUNT_ERRORS,
- ReplayColumn.ACTIVITY,
- ]}
- showDropdownFilters={false}
- />
- </Layout.Main>
- );
- }
- export default TransactionReplays;
|