123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796 |
- import {Fragment, useCallback, useEffect, useState} from 'react';
- import styled from '@emotion/styled';
- import * as Sentry from '@sentry/react';
- import * as qs from 'query-string';
- import FloatingFeedbackWidget from 'sentry/components/feedback/widget/floatingFeedbackWidget';
- import useDrawer from 'sentry/components/globalDrawer';
- import LoadingError from 'sentry/components/loadingError';
- import LoadingIndicator from 'sentry/components/loadingIndicator';
- import PageFiltersContainer from 'sentry/components/organizations/pageFilters/container';
- import MissingProjectMembership from 'sentry/components/projects/missingProjectMembership';
- import SentryDocumentTitle from 'sentry/components/sentryDocumentTitle';
- import {TabPanels, Tabs} from 'sentry/components/tabs';
- import {t} from 'sentry/locale';
- import GroupStore from 'sentry/stores/groupStore';
- import {space} from 'sentry/styles/space';
- import type {Event} from 'sentry/types/event';
- import type {Group} from 'sentry/types/group';
- import {GroupStatus, IssueCategory, IssueType} from 'sentry/types/group';
- import type {RouteComponentProps} from 'sentry/types/legacyReactRouter';
- import type {Organization} from 'sentry/types/organization';
- import type {Project} from 'sentry/types/project';
- import {defined} from 'sentry/utils';
- import {trackAnalytics} from 'sentry/utils/analytics';
- import {browserHistory} from 'sentry/utils/browserHistory';
- import {getUtcDateString} from 'sentry/utils/dates';
- import {
- getAnalyticsDataForEvent,
- getAnalyticsDataForGroup,
- getMessage,
- getTitle,
- } from 'sentry/utils/events';
- import {getConfigForIssueType} from 'sentry/utils/issueTypeConfig';
- import {getAnalyicsDataForProject} from 'sentry/utils/projects';
- import {setApiQueryData, useQueryClient} from 'sentry/utils/queryClient';
- import recreateRoute from 'sentry/utils/recreateRoute';
- import useDisableRouteAnalytics from 'sentry/utils/routeAnalytics/useDisableRouteAnalytics';
- import useRouteAnalyticsEventNames from 'sentry/utils/routeAnalytics/useRouteAnalyticsEventNames';
- import useRouteAnalyticsParams from 'sentry/utils/routeAnalytics/useRouteAnalyticsParams';
- import useApi from 'sentry/utils/useApi';
- import {useDetailedProject} from 'sentry/utils/useDetailedProject';
- import {useLocation} from 'sentry/utils/useLocation';
- import {useMemoWithPrevious} from 'sentry/utils/useMemoWithPrevious';
- import {useNavigate} from 'sentry/utils/useNavigate';
- import useOrganization from 'sentry/utils/useOrganization';
- import {useParams} from 'sentry/utils/useParams';
- import useProjects from 'sentry/utils/useProjects';
- import useRouter from 'sentry/utils/useRouter';
- import {useUser} from 'sentry/utils/useUser';
- import {ERROR_TYPES} from 'sentry/views/issueDetails/constants';
- import GroupEventDetails from 'sentry/views/issueDetails/groupEventDetails/groupEventDetails';
- import {useGroupTagsDrawer} from 'sentry/views/issueDetails/groupTags/useGroupTagsDrawer';
- import GroupHeader from 'sentry/views/issueDetails/header';
- import SampleEventAlert from 'sentry/views/issueDetails/sampleEventAlert';
- import {GroupDetailsLayout} from 'sentry/views/issueDetails/streamline/groupDetailsLayout';
- import {useMergedIssuesDrawer} from 'sentry/views/issueDetails/streamline/useMergedIssuesDrawer';
- import {useSimilarIssuesDrawer} from 'sentry/views/issueDetails/streamline/useSimilarIssuesDrawer';
- import {Tab} from 'sentry/views/issueDetails/types';
- import {makeFetchGroupQueryKey, useGroup} from 'sentry/views/issueDetails/useGroup';
- import {useGroupDetailsRoute} from 'sentry/views/issueDetails/useGroupDetailsRoute';
- import {useGroupEvent} from 'sentry/views/issueDetails/useGroupEvent';
- import {
- getGroupReprocessingStatus,
- markEventSeen,
- ReprocessingStatus,
- useDefaultIssueEvent,
- useEnvironmentsFromUrl,
- useHasStreamlinedUI,
- useIsSampleEvent,
- } from 'sentry/views/issueDetails/utils';
- type Error = (typeof ERROR_TYPES)[keyof typeof ERROR_TYPES] | null;
- type RouterParams = {groupId: string; eventId?: string};
- type RouteProps = RouteComponentProps<RouterParams, {}>;
- interface GroupDetailsProps extends RouteComponentProps<{groupId: string}, {}> {
- children: React.ReactNode;
- }
- type FetchGroupDetailsState = {
- error: boolean;
- errorType: Error;
- event: Event | null;
- group: Group | null;
- loadingGroup: boolean;
- refetchData: () => void;
- refetchGroup: () => void;
- };
- interface GroupDetailsContentProps extends GroupDetailsProps, FetchGroupDetailsState {
- group: Group;
- project: Project;
- }
- function getFetchDataRequestErrorType(status?: number | null): Error {
- if (!status) {
- return null;
- }
- if (status === 404) {
- return ERROR_TYPES.GROUP_NOT_FOUND;
- }
- if (status === 403) {
- return ERROR_TYPES.MISSING_MEMBERSHIP;
- }
- return null;
- }
- function getReprocessingNewRoute({
- group,
- currentTab,
- router,
- baseUrl,
- }: {
- baseUrl: string;
- currentTab: Tab;
- group: Group;
- router: RouteProps['router'];
- }) {
- const {routes, params, location} = router;
- const {groupId} = params;
- const {id: nextGroupId} = group;
- const reprocessingStatus = getGroupReprocessingStatus(group);
- if (groupId !== nextGroupId) {
- // Redirects to the Activities tab
- if (
- reprocessingStatus === ReprocessingStatus.REPROCESSED_AND_HASNT_EVENT &&
- currentTab !== Tab.ACTIVITY
- ) {
- return {
- pathname: `${baseUrl}${Tab.ACTIVITY}/`,
- query: {...params, groupId: nextGroupId},
- };
- }
- return recreateRoute('', {
- routes,
- location,
- params: {...params, groupId: nextGroupId},
- });
- }
- if (
- reprocessingStatus === ReprocessingStatus.REPROCESSING &&
- currentTab !== Tab.DETAILS
- ) {
- return {
- pathname: baseUrl,
- query: params,
- };
- }
- if (
- reprocessingStatus === ReprocessingStatus.REPROCESSED_AND_HASNT_EVENT &&
- currentTab !== Tab.ACTIVITY &&
- currentTab !== Tab.USER_FEEDBACK
- ) {
- return {
- pathname: `${baseUrl}${Tab.ACTIVITY}/`,
- query: params,
- };
- }
- return undefined;
- }
- function useRefetchGroupForReprocessing({
- refetchGroup,
- }: Pick<FetchGroupDetailsState, 'refetchGroup'>) {
- useEffect(() => {
- const refetchInterval = window.setInterval(refetchGroup, 30000);
- return () => {
- window.clearInterval(refetchInterval);
- };
- }, [refetchGroup]);
- }
- /**
- * This is a temporary measure to ensure that the GroupStore and query cache
- * are both up to date while we are still using both in the issue details page.
- * Once we remove all references to GroupStore in the issue details page we
- * should remove this.
- */
- function useSyncGroupStore(groupId: string, incomingEnvs: string[]) {
- const queryClient = useQueryClient();
- const organization = useOrganization();
- // It's possible the overview page is still unloading the store
- useEffect(() => {
- return GroupStore.listen(() => {
- const [storeGroup] = GroupStore.getState();
- if (
- defined(storeGroup) &&
- storeGroup.id === groupId &&
- // Check for properties that are only set after the group has been loaded
- defined(storeGroup.participants) &&
- defined(storeGroup.activity)
- ) {
- setApiQueryData(
- queryClient,
- makeFetchGroupQueryKey({
- groupId: storeGroup.id,
- organizationSlug: organization.slug,
- environments: incomingEnvs,
- }),
- storeGroup
- );
- }
- }, undefined) as () => void;
- }, [groupId, incomingEnvs, organization.slug, queryClient]);
- }
- function useFetchGroupDetails(): FetchGroupDetailsState {
- const api = useApi();
- const organization = useOrganization();
- const router = useRouter();
- const params = router.params;
- const navigate = useNavigate();
- const defaultIssueEvent = useDefaultIssueEvent();
- const [allProjectChanged, setAllProjectChanged] = useState<boolean>(false);
- const {currentTab, baseUrl} = useGroupDetailsRoute();
- const environments = useEnvironmentsFromUrl();
- const groupId = params.groupId;
- const {
- data: event,
- isPending: loadingEvent,
- isError: isEventError,
- refetch: refetchEvent,
- } = useGroupEvent({
- groupId,
- eventId: params.eventId,
- environments,
- });
- const {
- data: groupData,
- isPending: loadingGroup,
- isError: isGroupError,
- error: groupError,
- refetch: refetchGroupCall,
- } = useGroup({groupId});
- useEffect(() => {
- const eventIdUrl = params.eventId ?? defaultIssueEvent;
- const isLatestOrRecommendedEvent =
- eventIdUrl === 'latest' || eventIdUrl === 'recommended';
- if (isLatestOrRecommendedEvent && isEventError && router.location.query.query) {
- // If we get an error from the helpful event endpoint, it probably means
- // the query failed validation. We should remove the query to try again.
- navigate(
- {
- ...router.location,
- query: {
- ...router.location.query,
- query: undefined,
- },
- },
- {replace: true}
- );
- }
- }, [defaultIssueEvent, isEventError, navigate, router.location, params.eventId]);
- /**
- * Allows the GroupEventHeader to display the previous event while the new event is loading.
- * This is not closer to the GroupEventHeader because it is unmounted
- * between route changes like latest event => eventId
- */
- const previousEvent = useMemoWithPrevious<typeof event | null>(
- previousInstance => {
- if (event) {
- return event;
- }
- return previousInstance;
- },
- [event]
- );
- const group = groupData ?? null;
- useEffect(() => {
- if (defined(group)) {
- GroupStore.loadInitialData([group]);
- }
- }, [groupId, group]);
- useSyncGroupStore(groupId, environments);
- useEffect(() => {
- if (group && event) {
- const reprocessingNewRoute = getReprocessingNewRoute({
- group,
- currentTab,
- router,
- baseUrl,
- });
- if (reprocessingNewRoute) {
- browserHistory.push(reprocessingNewRoute);
- }
- }
- }, [group, event, router, currentTab, baseUrl]);
- useEffect(() => {
- const matchingProjectSlug = group?.project?.slug;
- if (!matchingProjectSlug) {
- return;
- }
- if (!group.hasSeen) {
- markEventSeen(api, organization.slug, matchingProjectSlug, params.groupId);
- }
- }, [
- api,
- group?.hasSeen,
- group?.project?.id,
- group?.project?.slug,
- organization.slug,
- params.groupId,
- ]);
- const allProjectsFlag = router.location.query._allp;
- useEffect(() => {
- const locationQuery = qs.parse(window.location.search) || {};
- // We use _allp as a temporary measure to know they came from the
- // issue list page with no project selected (all projects included in
- // filter).
- //
- // If it is not defined, we add the locked project id to the URL
- // (this is because if someone navigates directly to an issue on
- // single-project priveleges, then goes back - they were getting
- // assigned to the first project).
- //
- // If it is defined, we do not so that our back button will bring us
- // to the issue list page with no project selected instead of the
- // locked project.
- if (
- locationQuery.project === undefined &&
- !allProjectsFlag &&
- !allProjectChanged &&
- group?.project.id
- ) {
- locationQuery.project = group?.project.id;
- browserHistory.replace({...window.location, query: locationQuery});
- }
- if (allProjectsFlag && !allProjectChanged) {
- delete locationQuery.project;
- // We delete _allp from the URL to keep the hack a bit cleaner, but
- // this is not an ideal solution and will ultimately be replaced with
- // something smarter.
- delete locationQuery._allp;
- browserHistory.replace({...window.location, query: locationQuery});
- setAllProjectChanged(true);
- }
- }, [allProjectsFlag, group?.project.id, allProjectChanged]);
- const errorType = groupError ? getFetchDataRequestErrorType(groupError.status) : null;
- useEffect(() => {
- if (isGroupError) {
- Sentry.captureException(groupError);
- }
- }, [isGroupError, groupError]);
- const refetchGroup = useCallback(() => {
- if (group?.status !== GroupStatus.REPROCESSING || loadingGroup || loadingEvent) {
- return;
- }
- refetchGroupCall();
- }, [group, loadingGroup, loadingEvent, refetchGroupCall]);
- const refetchData = useCallback(() => {
- refetchEvent();
- refetchGroup();
- }, [refetchGroup, refetchEvent]);
- // Refetch when group is stale
- useEffect(() => {
- if (group && (group as Group & {stale?: boolean}).stale) {
- refetchGroup();
- }
- }, [refetchGroup, group]);
- useRefetchGroupForReprocessing({refetchGroup});
- useEffect(() => {
- return () => {
- GroupStore.reset();
- };
- }, []);
- return {
- loadingGroup,
- group,
- // Allow previous event to be displayed while new event is loading
- event: (loadingEvent ? event ?? previousEvent : event) ?? null,
- errorType,
- error: isGroupError,
- refetchData,
- refetchGroup,
- };
- }
- function useLoadedEventType() {
- const params = useParams<{eventId?: string}>();
- const defaultIssueEvent = useDefaultIssueEvent();
- switch (params.eventId) {
- case undefined:
- return defaultIssueEvent;
- case 'latest':
- case 'oldest':
- return params.eventId;
- default:
- return 'event_id';
- }
- }
- function useTrackView({
- group,
- event,
- project,
- tab,
- }: {
- event: Event | null;
- group: Group | null;
- tab: Tab;
- project?: Project;
- }) {
- const location = useLocation();
- const {alert_date, alert_rule_id, alert_type, ref_fallback, stream_index, query} =
- location.query;
- const groupEventType = useLoadedEventType();
- const user = useUser();
- useRouteAnalyticsEventNames('issue_details.viewed', 'Issue Details: Viewed');
- useRouteAnalyticsParams({
- ...getAnalyticsDataForGroup(group),
- ...getAnalyticsDataForEvent(event),
- ...getAnalyicsDataForProject(project),
- tab,
- stream_index: typeof stream_index === 'string' ? Number(stream_index) : undefined,
- query: typeof query === 'string' ? query : undefined,
- // Alert properties track if the user came from email/slack alerts
- alert_date:
- typeof alert_date === 'string' ? getUtcDateString(Number(alert_date)) : undefined,
- alert_rule_id: typeof alert_rule_id === 'string' ? alert_rule_id : undefined,
- alert_type: typeof alert_type === 'string' ? alert_type : undefined,
- ref_fallback,
- group_event_type: groupEventType,
- prefers_streamlined_ui: user?.options?.prefersIssueDetailsStreamlinedUI ?? false,
- });
- // Set default values for properties that may be updated in subcomponents.
- // Must be separate from the above values, otherwise the actual values filled in
- // by subcomponents may be overwritten when the above values change.
- useRouteAnalyticsParams({
- // Will be updated by StacktraceLink if there is a stacktrace link
- stacktrace_link_viewed: false,
- // Will be updated by IssueQuickTrace if there is a trace
- trace_status: 'none',
- // Will be updated in GroupDetailsHeader if there are replays
- group_has_replay: false,
- // Will be updated in ReplayPreview if there is a replay
- event_replay_status: 'none',
- // Will be updated in SuspectCommits if there are suspect commits
- num_suspect_commits: 0,
- suspect_commit_calculation: 'no suspect commit',
- // Will be updated in Autofix if enabled
- autofix_status: 'none',
- });
- useDisableRouteAnalytics(!group || !event || !project);
- }
- const trackTabChanged = ({
- organization,
- project,
- group,
- event,
- tab,
- }: {
- event: Event | null;
- group: Group;
- organization: Organization;
- project: Project;
- tab: Tab;
- }) => {
- if (!project || !group) {
- return;
- }
- trackAnalytics('issue_details.tab_changed', {
- organization,
- project_id: parseInt(project.id, 10),
- tab,
- ...getAnalyticsDataForGroup(group),
- });
- if (group.issueCategory !== IssueCategory.ERROR) {
- return;
- }
- const analyticsData = event
- ? event.tags
- .filter(({key}) => ['device', 'os', 'browser'].includes(key))
- .reduce((acc, {key, value}) => {
- acc[key] = value;
- return acc;
- }, {})
- : {};
- trackAnalytics('issue_group_details.tab.clicked', {
- organization,
- tab,
- platform: project.platform,
- ...analyticsData,
- });
- };
- function GroupDetailsContentError({
- errorType,
- onRetry,
- }: {
- errorType: Error;
- onRetry: () => void;
- }) {
- const organization = useOrganization();
- const location = useLocation();
- const projectId = location.query.project;
- const {projects} = useProjects();
- const project = projects.find(proj => proj.id === projectId);
- switch (errorType) {
- case ERROR_TYPES.GROUP_NOT_FOUND:
- return (
- <StyledLoadingError
- message={t('The issue you were looking for was not found.')}
- />
- );
- case ERROR_TYPES.MISSING_MEMBERSHIP:
- return <MissingProjectMembership organization={organization} project={project} />;
- default:
- return <StyledLoadingError onRetry={onRetry} />;
- }
- }
- function GroupDetailsContent({
- children,
- group,
- project,
- event,
- }: GroupDetailsContentProps) {
- const organization = useOrganization();
- const {openTagsDrawer} = useGroupTagsDrawer({group});
- const {openSimilarIssuesDrawer} = useSimilarIssuesDrawer({group, project});
- const {openMergedIssuesDrawer} = useMergedIssuesDrawer({group, project});
- const {isDrawerOpen} = useDrawer();
- const {currentTab, baseUrl} = useGroupDetailsRoute();
- const hasStreamlinedUI = useHasStreamlinedUI();
- useEffect(() => {
- if (!hasStreamlinedUI || isDrawerOpen) {
- return;
- }
- if (currentTab === Tab.TAGS) {
- openTagsDrawer();
- } else if (currentTab === Tab.SIMILAR_ISSUES) {
- openSimilarIssuesDrawer();
- } else if (currentTab === Tab.MERGED) {
- openMergedIssuesDrawer();
- }
- }, [
- currentTab,
- hasStreamlinedUI,
- isDrawerOpen,
- openTagsDrawer,
- openSimilarIssuesDrawer,
- openMergedIssuesDrawer,
- ]);
- useTrackView({group, event, project, tab: currentTab});
- const isDisplayingEventDetails = [
- Tab.DETAILS,
- Tab.TAGS,
- Tab.SIMILAR_ISSUES,
- Tab.MERGED,
- ].includes(currentTab);
- return hasStreamlinedUI ? (
- <GroupDetailsLayout group={group} event={event ?? undefined} project={project}>
- {isDisplayingEventDetails ? (
- // The router displays a loading indicator when switching to any of these tabs
- // Avoid lazy loading spinner by force rendering the GroupEventDetails component
- <GroupEventDetails />
- ) : (
- children
- )}
- </GroupDetailsLayout>
- ) : (
- <Tabs
- value={currentTab}
- onChange={tab => trackTabChanged({tab, group, project, event, organization})}
- >
- <GroupHeader
- organization={organization}
- event={event}
- group={group}
- baseUrl={baseUrl}
- project={project as Project}
- />
- <GroupTabPanels>
- <TabPanels.Item key={currentTab}>{children}</TabPanels.Item>
- </GroupTabPanels>
- </Tabs>
- );
- }
- function GroupDetailsPageContent(props: GroupDetailsProps & FetchGroupDetailsState) {
- const projectSlug = props.group?.project?.slug;
- const api = useApi();
- const organization = useOrganization();
- const [injectedEvent, setInjectedEvent] = useState(null);
- const {
- projects,
- initiallyLoaded: projectsLoaded,
- fetchError: errorFetchingProjects,
- } = useProjects({slugs: projectSlug ? [projectSlug] : []});
- // Preload detailed project data for highlighted data section
- useDetailedProject(
- {
- orgSlug: organization.slug,
- projectSlug: projectSlug ?? '',
- },
- {enabled: !!projectSlug}
- );
- const project = projects.find(({slug}) => slug === projectSlug);
- const projectWithFallback = project ?? projects[0];
- const isRegressionIssue =
- props.group?.issueType === IssueType.PERFORMANCE_DURATION_REGRESSION ||
- props.group?.issueType === IssueType.PERFORMANCE_ENDPOINT_REGRESSION;
- useEffect(() => {
- if (props.group && projectsLoaded && !project) {
- Sentry.withScope(scope => {
- const projectIds = projects.map(item => item.id);
- scope.setContext('missingProject', {
- projectId: props.group?.project.id,
- availableProjects: projectIds,
- });
- scope.setFingerprint(['group-details-project-not-found']);
- Sentry.captureException(new Error('Project not found'));
- });
- }
- }, [props.group, project, projects, projectsLoaded]);
- useEffect(() => {
- const fetchLatestEvent = async () => {
- const event = await api.requestPromise(
- `/organizations/${organization.slug}/issues/${props.group?.id}/events/latest/`
- );
- setInjectedEvent(event);
- };
- if (isRegressionIssue && !defined(props.event)) {
- fetchLatestEvent();
- }
- }, [
- api,
- organization.slug,
- props.event,
- props.group,
- props.group?.id,
- isRegressionIssue,
- ]);
- if (props.error) {
- return (
- <GroupDetailsContentError errorType={props.errorType} onRetry={props.refetchData} />
- );
- }
- if (errorFetchingProjects) {
- return <StyledLoadingError message={t('Error loading the specified project')} />;
- }
- if (projectSlug && !errorFetchingProjects && projectsLoaded && !projectWithFallback) {
- return (
- <StyledLoadingError message={t('The project %s does not exist', projectSlug)} />
- );
- }
- const regressionIssueLoaded = defined(injectedEvent ?? props.event);
- if (
- !projectsLoaded ||
- !projectWithFallback ||
- !props.group ||
- (isRegressionIssue && !regressionIssueLoaded)
- ) {
- return <LoadingIndicator />;
- }
- return (
- <GroupDetailsContent
- {...props}
- project={projectWithFallback}
- group={props.group}
- event={props.event ?? injectedEvent}
- />
- );
- }
- function GroupDetails(props: GroupDetailsProps) {
- const organization = useOrganization();
- const {group, ...fetchGroupDetailsProps} = useFetchGroupDetails();
- const isSampleError = useIsSampleEvent();
- const getGroupDetailsTitle = () => {
- const defaultTitle = 'Sentry';
- if (!group) {
- return defaultTitle;
- }
- const {title} = getTitle(group);
- const message = getMessage(group);
- const eventDetails = `${organization.slug} — ${group.project.slug}`;
- if (title && message) {
- return `${title}: ${message} — ${eventDetails}`;
- }
- return `${title || message || defaultTitle} — ${eventDetails}`;
- };
- const config = group && getConfigForIssueType(group, group.project);
- return (
- <Fragment>
- {isSampleError && group && (
- <SampleEventAlert project={group.project} organization={organization} />
- )}
- <SentryDocumentTitle noSuffix title={getGroupDetailsTitle()}>
- <PageFiltersContainer
- skipLoadLastUsed
- forceProject={group?.project}
- shouldForceProject
- >
- {config?.showFeedbackWidget && <FloatingFeedbackWidget />}
- <GroupDetailsPageContent
- {...props}
- {...{
- group,
- ...fetchGroupDetailsProps,
- }}
- />
- </PageFiltersContainer>
- </SentryDocumentTitle>
- </Fragment>
- );
- }
- export default Sentry.withProfiler(GroupDetails);
- const StyledLoadingError = styled(LoadingError)`
- margin: ${space(2)};
- `;
- const GroupTabPanels = styled(TabPanels)`
- flex-grow: 1;
- display: flex;
- flex-direction: column;
- justify-content: stretch;
- `;
|