123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682 |
- import {cloneElement, Component, Fragment, isValidElement} from 'react';
- import {browserHistory, RouteComponentProps} from 'react-router';
- import styled from '@emotion/styled';
- import * as Sentry from '@sentry/react';
- import * as PropTypes from 'prop-types';
- import {fetchOrganizationEnvironments} from 'sentry/actionCreators/environments';
- import {Client} from 'sentry/api';
- 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 SentryTypes from 'sentry/sentryTypes';
- import GroupStore from 'sentry/stores/groupStore';
- import {space} from 'sentry/styles/space';
- import {AvatarProject, Group, IssueCategory, Organization, Project} from 'sentry/types';
- import {Event} from 'sentry/types/event';
- import {trackAnalytics} from 'sentry/utils/analytics';
- import {getUtcDateString} from 'sentry/utils/dates';
- import {
- getAnalyticsDataForEvent,
- getAnalyticsDataForGroup,
- getMessage,
- getTitle,
- } from 'sentry/utils/events';
- import Projects, {getAnalyicsDataForProject} from 'sentry/utils/projects';
- import recreateRoute from 'sentry/utils/recreateRoute';
- import withRouteAnalytics, {
- WithRouteAnalyticsProps,
- } from 'sentry/utils/routeAnalytics/withRouteAnalytics';
- import withApi from 'sentry/utils/withApi';
- import {normalizeUrl} from 'sentry/utils/withDomainRequired';
- import {ERROR_TYPES} from './constants';
- import GroupHeader from './header';
- import SampleEventAlert from './sampleEventAlert';
- import {Tab, TabPaths} from './types';
- import {
- fetchGroupEvent,
- getGroupReprocessingStatus,
- markEventSeen,
- ReprocessingStatus,
- } from './utils';
- type Error = (typeof ERROR_TYPES)[keyof typeof ERROR_TYPES] | null;
- type Props = {
- api: Client;
- children: React.ReactNode;
- environments: string[];
- isGlobalSelectionReady: boolean;
- organization: Organization;
- projects: Project[];
- } & WithRouteAnalyticsProps &
- RouteComponentProps<{groupId: string; eventId?: string}, {}>;
- type State = {
- error: boolean;
- errorType: Error;
- eventError: boolean;
- group: Group | null;
- loading: boolean;
- loadingEvent: boolean;
- loadingGroup: boolean;
- project: null | (Pick<Project, 'id' | 'slug'> & Partial<Pick<Project, 'platform'>>);
- event?: Event;
- };
- class GroupDetails extends Component<Props, State> {
- static childContextTypes = {
- group: SentryTypes.Group,
- location: PropTypes.object,
- };
- state = this.initialState;
- getChildContext() {
- return {
- group: this.state.group,
- location: this.props.location,
- };
- }
- componentDidMount() {
- this.fetchData(true);
- this.updateReprocessingProgress();
- // Fetch environments early - used in GroupEventDetailsContainer
- fetchOrganizationEnvironments(this.props.api, this.props.organization.slug);
- }
- componentDidUpdate(prevProps: Props) {
- const globalSelectionReadyChanged =
- prevProps.isGlobalSelectionReady !== this.props.isGlobalSelectionReady;
- if (
- globalSelectionReadyChanged ||
- prevProps.location.pathname !== this.props.location.pathname
- ) {
- // Skip tracking for other navigation events like switching events
- this.fetchData(globalSelectionReadyChanged);
- }
- }
- componentWillUnmount() {
- GroupStore.reset();
- this.listener?.();
- if (this.refetchInterval) {
- window.clearInterval(this.refetchInterval);
- }
- }
- refetchInterval: number | null = null;
- get initialState(): State {
- return {
- group: null,
- loading: true,
- loadingEvent: true,
- loadingGroup: true,
- error: false,
- eventError: false,
- errorType: null,
- project: null,
- };
- }
- trackView(project: Project) {
- const {group, event} = this.state;
- const {location} = this.props;
- const {alert_date, alert_rule_id, alert_type, ref_fallback, stream_index, query} =
- location.query;
- this.props.setEventNames('issue_details.viewed', 'Issue Details: Viewed');
- this.props.setRouteAnalyticsParams({
- ...getAnalyticsDataForGroup(group),
- ...getAnalyticsDataForEvent(event),
- ...getAnalyicsDataForProject(project),
- 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,
- // 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',
- });
- }
- remountComponent = () => {
- this.setState(this.initialState);
- this.fetchData();
- };
- get groupDetailsEndpoint() {
- return `/issues/${this.props.params.groupId}/`;
- }
- get groupReleaseEndpoint() {
- return `/issues/${this.props.params.groupId}/first-last-release/`;
- }
- async getEvent() {
- this.setState({loadingEvent: true, eventError: false});
- const {params, environments, api} = this.props;
- const groupId = params.groupId;
- const eventId = params.eventId ?? 'latest';
- try {
- const event = await fetchGroupEvent(api, groupId, eventId, environments);
- this.setState({event, loading: false, eventError: false, loadingEvent: false});
- } catch (err) {
- // This is an expected error, capture to Sentry so that it is not considered as an unhandled error
- Sentry.captureException(err);
- this.setState({eventError: true, loading: false, loadingEvent: false});
- }
- }
- getCurrentTab() {
- const {router, routes} = this.props;
- const currentRoute = routes[routes.length - 1];
- // If we're in the tag details page ("/tags/:tagKey/")
- if (router.params.tagKey) {
- return Tab.TAGS;
- }
- return (
- Object.values(Tab).find(tab => currentRoute.path === TabPaths[tab]) ?? Tab.DETAILS
- );
- }
- getCurrentRouteInfo(group: Group): {baseUrl: string; currentTab: Tab} {
- const {organization, params} = this.props;
- const {event} = this.state;
- const currentTab = this.getCurrentTab();
- const baseUrl = normalizeUrl(
- `/organizations/${organization.slug}/issues/${group.id}/${
- params.eventId && event ? `events/${event.id}/` : ''
- }`
- );
- return {baseUrl, currentTab};
- }
- updateReprocessingProgress() {
- const hasReprocessingV2Feature = this.hasReprocessingV2Feature();
- if (!hasReprocessingV2Feature) {
- return;
- }
- if (this.refetchInterval) {
- window.clearInterval(this.refetchInterval);
- }
- this.refetchInterval = window.setInterval(this.refetchGroup, 30000);
- }
- hasReprocessingV2Feature() {
- const {organization} = this.props;
- return organization.features?.includes('reprocessing-v2');
- }
- getReprocessingNewRoute(data: Group) {
- const {routes, location, params} = this.props;
- const {groupId} = params;
- const {id: nextGroupId} = data;
- const hasReprocessingV2Feature = this.hasReprocessingV2Feature();
- const reprocessingStatus = getGroupReprocessingStatus(data);
- const {currentTab, baseUrl} = this.getCurrentRouteInfo(data);
- if (groupId !== nextGroupId) {
- if (hasReprocessingV2Feature) {
- // 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 (hasReprocessingV2Feature) {
- 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;
- }
- getGroupQuery(): Record<string, string | string[]> {
- const {environments} = this.props;
- // Note, we do not want to include the environment key at all if there are no environments
- const query: Record<string, string | string[]> = {
- ...(environments ? {environment: environments} : {}),
- expand: ['inbox', 'owners'],
- collapse: 'release',
- };
- return query;
- }
- getFetchDataRequestErrorType(status: any): Error {
- if (!status) {
- return null;
- }
- if (status === 404) {
- return ERROR_TYPES.GROUP_NOT_FOUND;
- }
- if (status === 403) {
- return ERROR_TYPES.MISSING_MEMBERSHIP;
- }
- return null;
- }
- handleRequestError(error: any) {
- Sentry.captureException(error);
- const errorType = this.getFetchDataRequestErrorType(error?.status);
- this.setState({
- loadingGroup: false,
- loading: false,
- error: true,
- errorType,
- });
- }
- refetchGroup = async () => {
- const {loadingGroup, loading, loadingEvent, group} = this.state;
- if (
- group?.status !== ReprocessingStatus.REPROCESSING ||
- loadingGroup ||
- loading ||
- loadingEvent
- ) {
- return;
- }
- const {api} = this.props;
- this.setState({loadingGroup: true});
- try {
- const updatedGroup = await api.requestPromise(this.groupDetailsEndpoint, {
- query: this.getGroupQuery(),
- });
- const reprocessingNewRoute = this.getReprocessingNewRoute(updatedGroup);
- if (reprocessingNewRoute) {
- browserHistory.push(reprocessingNewRoute);
- return;
- }
- this.setState({group: updatedGroup, loadingGroup: false});
- } catch (error) {
- this.handleRequestError(error);
- }
- };
- async fetchGroupReleases() {
- const {api} = this.props;
- const releases = await api.requestPromise(this.groupReleaseEndpoint);
- GroupStore.onPopulateReleases(this.props.params.groupId, releases);
- }
- async fetchData(trackView = false) {
- const {api, isGlobalSelectionReady, organization, params} = this.props;
- // Need to wait for global selection store to be ready before making request
- if (!isGlobalSelectionReady) {
- return;
- }
- try {
- const eventPromise = this.getEvent();
- const groupPromise = await api.requestPromise(this.groupDetailsEndpoint, {
- query: this.getGroupQuery(),
- });
- const [data] = await Promise.all([groupPromise, eventPromise]);
- const groupReleasePromise = this.fetchGroupReleases();
- const reprocessingNewRoute = this.getReprocessingNewRoute(data);
- if (reprocessingNewRoute) {
- browserHistory.push(reprocessingNewRoute);
- return;
- }
- const project = this.props.projects.find(p => p.id === data.project.id);
- if (!project) {
- Sentry.withScope(scope => {
- const projectIds = this.props.projects.map(item => item.id);
- scope.setContext('missingProject', {
- projectId: data.project.id,
- availableProjects: projectIds,
- });
- Sentry.captureException(new Error('Project not found'));
- });
- } else {
- markEventSeen(api, organization.slug, project.slug, params.groupId);
- const locationWithProject = {...this.props.location};
- if (
- locationWithProject.query.project === undefined &&
- locationWithProject.query._allp === undefined
- ) {
- // 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.
- locationWithProject.query = {...locationWithProject.query, project: project.id};
- }
- // 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 locationWithProject.query._allp;
- browserHistory.replace(locationWithProject);
- }
- this.setState({project: project || data.project, loadingGroup: false});
- GroupStore.loadInitialData([data]);
- if (trackView) {
- // make sure releases have loaded before we track the view
- groupReleasePromise.then(() => project && this.trackView(project));
- }
- } catch (error) {
- this.handleRequestError(error);
- }
- }
- listener = GroupStore.listen(itemIds => this.onGroupChange(itemIds), undefined);
- onGroupChange(itemIds: Set<string>) {
- const id = this.props.params.groupId;
- if (itemIds.has(id)) {
- const group = GroupStore.get(id) as Group;
- if (group) {
- // TODO(ts) This needs a better approach. issueActions is splicing attributes onto
- // group objects to cheat here.
- if ((group as Group & {stale?: boolean}).stale) {
- this.fetchData();
- return;
- }
- this.setState({
- group,
- });
- }
- }
- }
- getTitle() {
- const {organization} = this.props;
- const {group} = this.state;
- const defaultTitle = 'Sentry';
- if (!group) {
- return defaultTitle;
- }
- const {title} = getTitle(group, organization?.features);
- const message = getMessage(group);
- const {project} = group;
- const eventDetails = `${organization.slug} - ${project.slug}`;
- if (title && message) {
- return `${title}: ${message} - ${eventDetails}`;
- }
- return `${title || message || defaultTitle} - ${eventDetails}`;
- }
- tabClickAnalyticsEvent(tab: Tab) {
- const {organization} = this.props;
- const {project, group, event} = this.state;
- 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,
- });
- }
- renderError() {
- const {projects, location} = this.props;
- const projectId = location.query.project;
- const project = projects.find(proj => proj.id === projectId);
- switch (this.state.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={this.props.organization}
- project={project}
- />
- );
- default:
- return <StyledLoadingError onRetry={this.remountComponent} />;
- }
- }
- renderContent(project: AvatarProject, group: Group) {
- const {children, environments, organization, router} = this.props;
- const {loadingEvent, eventError, event} = this.state;
- const {currentTab, baseUrl} = this.getCurrentRouteInfo(group);
- const groupReprocessingStatus = getGroupReprocessingStatus(group);
- let childProps: Record<string, any> = {
- environments,
- group,
- project,
- };
- if (currentTab === Tab.DETAILS) {
- if (group.id !== event?.groupID && !eventError) {
- // if user pastes only the event id into the url, but it's from another group, redirect to correct group/event
- const redirectUrl = `/organizations/${organization.slug}/issues/${event?.groupID}/events/${event?.id}/`;
- router.push(normalizeUrl(redirectUrl));
- } else {
- childProps = {
- ...childProps,
- event,
- loadingEvent,
- eventError,
- groupReprocessingStatus,
- onRetry: () => this.remountComponent(),
- };
- }
- }
- if (currentTab === Tab.TAGS) {
- childProps = {...childProps, event, baseUrl};
- }
- return (
- <Tabs value={currentTab} onChange={tab => this.tabClickAnalyticsEvent(tab)}>
- <GroupHeader
- organization={organization}
- groupReprocessingStatus={groupReprocessingStatus}
- event={event}
- group={group}
- baseUrl={baseUrl}
- project={project as Project}
- />
- <GroupTabPanels>
- <TabPanels.Item key={currentTab}>
- {isValidElement(children) ? cloneElement(children, childProps) : children}
- </TabPanels.Item>
- </GroupTabPanels>
- </Tabs>
- );
- }
- renderPageContent() {
- const {error: isError, group, project, loading} = this.state;
- const isLoading = loading || (!group && !isError);
- if (isLoading) {
- return <LoadingIndicator />;
- }
- if (isError) {
- return this.renderError();
- }
- const {organization} = this.props;
- return (
- <Projects
- orgId={organization.slug}
- slugs={[project?.slug ?? '']}
- data-test-id="group-projects-container"
- >
- {({projects, initiallyLoaded, fetchError}) =>
- initiallyLoaded ? (
- fetchError ? (
- <StyledLoadingError message={t('Error loading the specified project')} />
- ) : (
- // TODO(ts): Update renderContent function to deal with empty group
- // Search for the slug in the projects list if possible. This is because projects
- // is just a complete list of stored projects and the first element may not be
- // the expected project.
- this.renderContent(
- (project?.slug
- ? projects.find(({slug}) => slug === project?.slug)
- : undefined) ?? projects[0],
- group!
- )
- )
- ) : (
- <LoadingIndicator />
- )
- }
- </Projects>
- );
- }
- render() {
- const {project, group} = this.state;
- const {organization} = this.props;
- const isSampleError = group?.tags?.some(tag => tag.key === 'sample_event');
- return (
- <Fragment>
- {isSampleError && project && (
- <SampleEventAlert project={project} organization={organization} />
- )}
- <SentryDocumentTitle noSuffix title={this.getTitle()}>
- <PageFiltersContainer
- skipLoadLastUsed
- forceProject={project}
- shouldForceProject
- >
- {this.renderPageContent()}
- </PageFiltersContainer>
- </SentryDocumentTitle>
- </Fragment>
- );
- }
- }
- export default withRouteAnalytics(withApi(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;
- `;
|