import {Component, Fragment} from 'react'; import styled from '@emotion/styled'; import type {Location, Query} from 'history'; import * as qs from 'query-string'; import LoadingError from 'sentry/components/loadingError'; import LoadingIndicator from 'sentry/components/loadingIndicator'; import QueryCount from 'sentry/components/queryCount'; import {t, tct} from 'sentry/locale'; import type {Fingerprint} from 'sentry/stores/groupingStore'; import GroupingStore from 'sentry/stores/groupingStore'; import {space} from 'sentry/styles/space'; import type {Group} from 'sentry/types/group'; import type {Organization} from 'sentry/types/organization'; import type {Project} from 'sentry/types/project'; import {trackAnalytics} from 'sentry/utils/analytics'; import withOrganization from 'sentry/utils/withOrganization'; import MergedList from './mergedList'; type Props = { groupId: Group['id']; location: Location; organization: Organization; project: Project; }; type State = { error: boolean; loading: boolean; mergedItems: Array; query: string; mergedLinks?: string; }; class GroupMergedView extends Component { state: State = { mergedItems: [], loading: true, error: false, query: (this.props.location.query.query ?? '') as string, }; componentDidMount() { this.fetchData(); } UNSAFE_componentWillReceiveProps(nextProps: Props) { if ( nextProps.groupId !== this.props.groupId || nextProps.location.search !== this.props.location.search ) { this.setState( { query: (nextProps.location.query.query ?? '') as string, }, this.fetchData ); } } componentWillUnmount() { this.listener?.(); } onGroupingChange = ({mergedItems, mergedLinks, loading, error}) => { if (mergedItems) { this.setState({ mergedItems, mergedLinks, loading: typeof loading !== 'undefined' ? loading : false, error: typeof error !== 'undefined' ? error : false, }); } }; listener = GroupingStore.listen(this.onGroupingChange, undefined); getEndpoint() { const {groupId, location, organization} = this.props; const queryParams = { ...location.query, limit: 50, query: this.state.query, }; return `/organizations/${organization.slug}/issues/${groupId}/hashes/?${qs.stringify( queryParams )}`; } fetchData = () => { GroupingStore.onFetch([ { endpoint: this.getEndpoint(), dataKey: 'merged', queryParams: this.props.location.query, }, ]); }; handleUnmerge = () => { const {organization, groupId} = this.props; GroupingStore.onUnmerge({ groupId, orgSlug: organization.slug, loadingMessage: t('Unmerging events\u2026'), successMessage: t('Events successfully queued for unmerging.'), errorMessage: t('Unable to queue events for unmerging.'), }); const unmergeKeys = [...GroupingStore.getState().unmergeList.values()]; trackAnalytics('issue_details.merged_tab.unmerge_clicked', { organization, group_id: groupId, event_ids_unmerged: unmergeKeys.join(','), total_unmerged: unmergeKeys.length, }); }; render() { const {project, organization, groupId} = this.props; const {loading: isLoading, error, mergedItems, mergedLinks} = this.state; const isError = error && !isLoading; const isLoadedSuccessfully = !isError && !isLoading; const fingerprintsWithLatestEvent = mergedItems.filter( ({latestEvent}) => !!latestEvent ); return ( {tct('Fingerprints included in this issue [count]', { count: <QueryCount count={fingerprintsWithLatestEvent.length} />, })} { // TODO: Once clickhouse is upgraded and the lag is no longer an issue, revisit this wording. // See https://github.com/getsentry/sentry/issues/56334. t( 'This is an experimental feature. All changes may take up to 24 hours take effect.' ) } {isLoading && } {isError && ( )} {isLoadedSuccessfully && ( )} ); } } export default withOrganization(GroupMergedView); const Title = styled('h4')` font-size: ${p => p.theme.fontSizeLarge}; margin-bottom: ${space(0.75)}; `; const HeaderWrapper = styled('div')` margin-bottom: ${space(2)}; small { color: ${p => p.theme.subText}; } `;