import {Fragment, useEffect} from 'react'; import {browserHistory, RouteComponentProps} from 'react-router'; import styled from '@emotion/styled'; import * as Sentry from '@sentry/react'; import isEqual from 'lodash/isEqual'; import {fetchSentryAppComponents} from 'sentry/actionCreators/sentryAppComponents'; import {Client} from 'sentry/api'; import ArchivedBox from 'sentry/components/archivedBox'; import GroupEventDetailsLoadingError from 'sentry/components/errors/groupEventDetailsLoadingError'; import {withMeta} from 'sentry/components/events/meta/metaProxy'; import HookOrDefault from 'sentry/components/hookOrDefault'; import * as Layout from 'sentry/components/layouts/thirds'; import LoadingIndicator from 'sentry/components/loadingIndicator'; import MutedBox from 'sentry/components/mutedBox'; import {TransactionProfileIdProvider} from 'sentry/components/profiling/transactionProfileIdProvider'; import ResolutionBox from 'sentry/components/resolutionBox'; import {space} from 'sentry/styles/space'; import { Group, GroupActivityReprocess, GroupReprocessing, IssueType, Organization, Project, } from 'sentry/types'; import {Event} from 'sentry/types/event'; import {defined} from 'sentry/utils'; import fetchSentryAppInstallations from 'sentry/utils/fetchSentryAppInstallations'; import {QuickTraceContext} from 'sentry/utils/performance/quickTrace/quickTraceContext'; import QuickTraceQuery from 'sentry/utils/performance/quickTrace/quickTraceQuery'; import {VisuallyCompleteWithData} from 'sentry/utils/performanceForSentry'; import usePrevious from 'sentry/utils/usePrevious'; import {normalizeUrl} from 'sentry/utils/withDomainRequired'; import GroupEventDetailsContent from 'sentry/views/issueDetails/groupEventDetails/groupEventDetailsContent'; import GroupEventHeader from 'sentry/views/issueDetails/groupEventHeader'; import GroupSidebar from 'sentry/views/issueDetails/groupSidebar'; import ReprocessingProgress from '../reprocessingProgress'; import { getEventEnvironment, getGroupMostRecentActivity, ReprocessingStatus, useEnvironmentsFromUrl, } from '../utils'; const EscalatingIssuesFeedback = HookOrDefault({ hookName: 'component:escalating-issues-banner-feedback', }); export interface GroupEventDetailsProps extends RouteComponentProps<{groupId: string; eventId?: string}, {}> { api: Client; eventError: boolean; group: Group; groupReprocessingStatus: ReprocessingStatus; loadingEvent: boolean; onRetry: () => void; organization: Organization; project: Project; event?: Event; } function GroupEventDetails(props: GroupEventDetailsProps) { const { group, project, organization, location, event, groupReprocessingStatus, loadingEvent, onRetry, eventError, api, params, } = props; const eventWithMeta = withMeta(event); // Reprocessing const hasReprocessingV2Feature = organization.features?.includes('reprocessing-v2'); const {activity: activities} = group; const mostRecentActivity = getGroupMostRecentActivity(activities); const orgSlug = organization.slug; const projectId = project.id; const environments = useEnvironmentsFromUrl(); const prevEnvironment = usePrevious(environments); const prevEvent = usePrevious(event); // load the data useEffect(() => { fetchSentryAppInstallations(api, orgSlug); // TODO(marcos): Sometimes PageFiltersStore cannot pick a project. if (projectId) { fetchSentryAppComponents(api, orgSlug, projectId); } else { Sentry.withScope(scope => { scope.setExtra('orgSlug', orgSlug); scope.setExtra('projectId', projectId); Sentry.captureMessage('Project ID was not set'); }); } }, [api, orgSlug, projectId]); // If environments are being actively changed and will no longer contain the // current event's environment, redirect to latest useEffect(() => { const environmentsHaveChanged = !isEqual(prevEnvironment, environments); // If environments are being actively changed and will no longer contain the // current event's environment, redirect to latest if ( environmentsHaveChanged && prevEvent && params.eventId && !['latest', 'oldest'].includes(params.eventId) ) { const shouldRedirect = environments.length > 0 && !environments.find(env => env === getEventEnvironment(prevEvent as Event)); if (shouldRedirect) { browserHistory.replace( normalizeUrl({ pathname: `/organizations/${organization.slug}/issues/${params.groupId}/`, query: location.query, }) ); return; } } }, [ prevEnvironment, environments, location.query, organization.slug, params, prevEvent, ]); const renderGroupStatusBanner = () => { const hasEscalatingIssuesUi = organization.features.includes('escalating-issues'); if (group.status === 'ignored') { return ( {hasEscalatingIssuesUi ? ( ) : ( )} ); } if (group.status === 'resolved') { return ( ); } return null; }; const renderContent = () => { if (loadingEvent) { return ; } if (eventError) { return ( ); } return ( ); }; return ( {hasReprocessingV2Feature && groupReprocessingStatus === ReprocessingStatus.REPROCESSING ? ( ) : ( {results => { return ( {renderGroupStatusBanner()} {eventWithMeta && group.issueType !== IssueType.PERFORMANCE_DURATION_REGRESSION && ( )} {renderContent()} ); }} )} ); } const StyledLayoutBody = styled(Layout.Body)` /* Makes the borders align correctly */ padding: 0 !important; @media (min-width: ${p => p.theme.breakpoints.large}) { align-content: stretch; } `; const GroupStatusBannerWrapper = styled('div')` margin-bottom: ${space(2)}; `; const StyledLayoutMain = styled(Layout.Main)` padding-top: ${space(2)}; @media (max-width: ${p => p.theme.breakpoints.medium}) { padding-top: ${space(1)}; } @media (min-width: ${p => p.theme.breakpoints.large}) { border-right: 1px solid ${p => p.theme.border}; padding-right: 0; } `; const StyledLayoutSide = styled(Layout.Side)` padding: ${space(3)} ${space(2)} ${space(3)}; @media (min-width: ${p => p.theme.breakpoints.large}) { padding-right: ${space(4)}; padding-left: 0; } `; export default GroupEventDetails;