123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749 |
- 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 omit from 'lodash/omit';
- import * as PropTypes from 'prop-types';
- 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 {Item, 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,
- GroupActivityAssigned,
- GroupActivityType,
- IssueCategory,
- Organization,
- Project,
- } from 'sentry/types';
- import {Event} from 'sentry/types/event';
- import trackAdvancedAnalyticsEvent from 'sentry/utils/analytics/trackAdvancedAnalyticsEvent';
- import {getUtcDateString} from 'sentry/utils/dates';
- import {TableData} from 'sentry/utils/discover/discoverQuery';
- import EventView from 'sentry/utils/discover/eventView';
- import {doDiscoverQuery} from 'sentry/utils/discover/genericDiscoverQuery';
- import {getAnalyicsDataForEvent, getMessage, getTitle} from 'sentry/utils/events';
- import getDaysSinceDate from 'sentry/utils/getDaysSinceDate';
- import Projects from 'sentry/utils/projects';
- import recreateRoute from 'sentry/utils/recreateRoute';
- import withApi from 'sentry/utils/withApi';
- 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';
- /**
- * Return the integration type for the first assignment via integration
- */
- function getAssignmentIntegration(group: Group) {
- if (!group.activity) {
- return '';
- }
- const assignmentAcitivies = group.activity.filter(
- activity => activity.type === GroupActivityType.ASSIGNED
- ) as GroupActivityAssigned[];
- const integrationAssignments = assignmentAcitivies.find(
- activity => !!activity.data.integration
- );
- return integrationAssignments?.data.integration || '';
- }
- 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[];
- } & RouteComponentProps<{groupId: string; orgId: string; eventId?: string}, {}>;
- type State = {
- error: boolean;
- errorType: Error;
- eventError: boolean;
- group: Group | null;
- loading: boolean;
- loadingEvent: boolean;
- loadingGroup: boolean;
- loadingReplayIds: boolean;
- project: null | (Pick<Project, 'id' | 'slug'> & Partial<Pick<Project, 'platform'>>);
- replayIds: null | string[];
- 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() {
- // only track the view if we are loading the event early
- this.fetchData(this.canLoadEventEarly(this.props));
- if (this.props.organization.features.includes('session-replay-ui')) {
- this.fetchReplayIds();
- }
- this.updateReprocessingProgress();
- }
- componentDidUpdate(prevProps: Props, prevState: State) {
- 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 && this.canLoadEventEarly(this.props));
- }
- if (
- (!this.canLoadEventEarly(prevProps) && !prevState?.group && this.state.group) ||
- (prevProps.params?.eventId !== this.props.params?.eventId && this.state.group)
- ) {
- // if we are loading events we should record analytics after it's loaded
- this.getEvent(this.state.group).then(
- () => this.state.group?.project && this.trackView(this.state.group?.project)
- );
- }
- }
- componentWillUnmount() {
- GroupStore.reset();
- this.listener?.();
- if (this.refetchInterval) {
- window.clearInterval(this.refetchInterval);
- }
- }
- refetchInterval: number | null = null;
- get initialState(): State {
- return {
- group: null,
- loading: true,
- loadingReplayIds: true,
- loadingEvent: true,
- loadingGroup: true,
- error: false,
- eventError: false,
- errorType: null,
- project: null,
- replayIds: null,
- };
- }
- trackView(project: Project) {
- const {group, event} = this.state;
- const {organization, params, location} = this.props;
- const {alert_date, alert_rule_id, alert_type} = location.query;
- trackAdvancedAnalyticsEvent('issue_details.viewed', {
- organization,
- project_id: parseInt(project.id, 10),
- group_id: parseInt(params.groupId, 10),
- // group properties
- issue_category: group?.issueCategory ?? IssueCategory.ERROR,
- issue_status: group?.status,
- issue_age: group?.firstSeen ? getDaysSinceDate(group.firstSeen) : -1,
- issue_level: group?.level,
- is_assigned: !!group?.assignedTo,
- error_count: Number(group?.count || -1),
- num_comments: group ? group.numComments : -1,
- project_platform: group?.project.platform,
- has_external_issue: group?.annotations ? group?.annotations.length > 0 : false,
- has_owner: group?.owners ? group?.owners.length > 0 : false,
- integration_assignment_source: group ? getAssignmentIntegration(group) : '',
- // event properties
- ...getAnalyicsDataForEvent(event),
- // 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,
- });
- }
- remountComponent = () => {
- this.setState(this.initialState);
- this.fetchData();
- };
- canLoadEventEarly(props: Props) {
- return !props.params.eventId || ['oldest', 'latest'].includes(props.params.eventId);
- }
- get groupDetailsEndpoint() {
- return `/issues/${this.props.params.groupId}/`;
- }
- get groupReleaseEndpoint() {
- return `/issues/${this.props.params.groupId}/first-last-release/`;
- }
- async getEvent(group?: Group) {
- if (group) {
- this.setState({loadingEvent: true, eventError: false});
- }
- const {params, environments, api} = this.props;
- const orgSlug = params.orgId;
- const groupId = params.groupId;
- const eventId = params.eventId ?? 'latest';
- const projectId = group?.project?.slug;
- try {
- const event = await fetchGroupEvent(
- api,
- orgSlug,
- groupId,
- eventId,
- environments,
- projectId
- );
- 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});
- }
- }
- getCurrentRouteInfo(group: Group): {baseUrl: string; currentTab: Tab} {
- const {organization, router, routes} = this.props;
- const {event} = this.state;
- const currentRoute = routes[routes.length - 1];
- let currentTab: Tab;
- // If we're in the tag details page ("/tags/:tagKey/")
- if (router.params.tagKey) {
- currentTab = Tab.TAGS;
- } else {
- currentTab =
- Object.values(Tab).find(tab => currentRoute.path === TabPaths[tab]) ??
- Tab.DETAILS;
- }
- const baseUrl = `/organizations/${organization.slug}/issues/${group.id}/${
- router.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 fetchReplayIds() {
- const {api, location, organization, params} = this.props;
- const {groupId} = params;
- this.setState({loadingReplayIds: true});
- const eventView = EventView.fromNewQueryWithLocation(
- {
- id: '',
- name: `Errors within replay`,
- version: 2,
- fields: ['replayId', 'count()'],
- query: `issue.id:${groupId} !replayId:""`,
- projects: [],
- },
- location
- );
- try {
- const [data] = await doDiscoverQuery<TableData>(
- api,
- `/organizations/${organization.slug}/events/`,
- eventView.getEventsAPIPayload(location)
- );
- const replayIds = data.data.map(record => String(record.replayId));
- this.setState({
- replayIds,
- loadingReplayIds: false,
- });
- } catch (err) {
- this.setState({loadingReplayIds: false});
- }
- }
- async fetchData(trackView = false) {
- const {api, isGlobalSelectionReady, params} = this.props;
- // Need to wait for global selection store to be ready before making request
- if (!isGlobalSelectionReady) {
- return;
- }
- try {
- const eventPromise = this.canLoadEventEarly(this.props)
- ? this.getEvent()
- : undefined;
- 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 = data.project;
- markEventSeen(api, params.orgId, project.slug, params.groupId);
- if (!project) {
- Sentry.withScope(() => {
- Sentry.captureException(new Error('Project not found'));
- });
- } else {
- 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, loadingGroup: false});
- GroupStore.loadInitialData([data]);
- if (trackView) {
- // make sure releases have loaded before we track the view
- groupReleasePromise.then(() => 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} = this.state;
- if (!project || !group) {
- return;
- }
- trackAdvancedAnalyticsEvent('issue_details.tab_changed', {
- organization,
- group_id: parseInt(group.id, 10),
- issue_category: group.issueCategory,
- project_id: parseInt(project.id, 10),
- tab,
- });
- }
- 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, location, router} = this.props;
- const {loadingEvent, eventError, event, replayIds} = 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(redirectUrl);
- } else {
- childProps = {
- ...childProps,
- event,
- loadingEvent,
- eventError,
- groupReprocessingStatus,
- onRetry: () => this.remountComponent(),
- };
- }
- }
- if (currentTab === Tab.TAGS) {
- childProps = {...childProps, event, baseUrl};
- } else if (currentTab === Tab.REPLAYS) {
- childProps = {...childProps, replayIds};
- }
- return (
- <Tabs
- value={currentTab}
- onChange={tab => {
- this.tabClickAnalyticsEvent(tab);
- router.push({
- pathname: `${baseUrl}${TabPaths[tab]}`,
- query: tab === Tab.EVENTS ? omit(location.query, 'query') : location.query,
- });
- }}
- >
- <GroupHeader
- organization={organization}
- groupReprocessingStatus={groupReprocessingStatus}
- event={event}
- group={group}
- replaysCount={replayIds?.length}
- baseUrl={baseUrl}
- project={project as Project}
- />
- <GroupTabPanels>
- <Item key={currentTab}>
- {isValidElement(children) ? cloneElement(children, childProps) : children}
- </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
- this.renderContent(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 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;
- `;
|