123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451 |
- import {Fragment, useEffect, useState} from 'react';
- import {browserHistory, InjectedRouter} from 'react-router';
- import styled from '@emotion/styled';
- import {Location} from 'history';
- import debounce from 'lodash/debounce';
- import {Client} from 'sentry/api';
- import Button from 'sentry/components/button';
- import ButtonBar from 'sentry/components/buttonBar';
- import {FeatureFeedback} from 'sentry/components/featureFeedback';
- import RangeSlider from 'sentry/components/forms/controls/rangeSlider';
- import Slider from 'sentry/components/forms/controls/rangeSlider/slider';
- import * as Layout from 'sentry/components/layouts/thirds';
- import LoadingIndicator from 'sentry/components/loadingIndicator';
- import Pagination from 'sentry/components/pagination';
- import {PanelTable} from 'sentry/components/panels';
- import {DEFAULT_DEBOUNCE_DURATION} from 'sentry/constants';
- import {t, tct, tn} from 'sentry/locale';
- import space from 'sentry/styles/space';
- import {BaseGroup, Group, Organization, Project} from 'sentry/types';
- import {defined} from 'sentry/utils';
- import parseLinkHeader from 'sentry/utils/parseLinkHeader';
- import withApi from 'sentry/utils/withApi';
- import ErrorMessage from './errorMessage';
- import NewIssue from './newIssue';
- type Error = React.ComponentProps<typeof ErrorMessage>['error'];
- type Props = {
- api: Client;
- groupId: Group['id'];
- location: Location<{cursor?: string; level?: number}>;
- organization: Organization;
- projSlug: Project['slug'];
- router: InjectedRouter;
- };
- type GroupingLevelDetails = Partial<Pick<BaseGroup, 'title' | 'metadata'>> & {
- eventCount: number;
- hash: string;
- latestEvent: BaseGroup['latestEvent'];
- };
- type GroupingLevel = {
- id: number;
- isCurrent: boolean;
- };
- export const groupingFeedbackTypes = [
- t('Too eager grouping'),
- t('Too specific grouping'),
- t('Other grouping issue'),
- ];
- const GROUPING_BREAKDOWN__DOC_LINK =
- 'https://docs.sentry.io/product/data-management-settings/event-grouping/grouping-breakdown/';
- function Grouping({api, groupId, location, organization, router, projSlug}: Props) {
- const {cursor, level} = location.query;
- const [isLoading, setIsLoading] = useState(false);
- const [isGroupingLevelDetailsLoading, setIsGroupingLevelDetailsLoading] =
- useState(false);
- const [error, setError] = useState<undefined | Error | string>(undefined);
- const [groupingLevels, setGroupingLevels] = useState<GroupingLevel[]>([]);
- const [activeGroupingLevel, setActiveGroupingLevel] = useState<number | undefined>(
- undefined
- );
- const [activeGroupingLevelDetails, setActiveGroupingLevelDetails] = useState<
- GroupingLevelDetails[]
- >([]);
- const [pagination, setPagination] = useState('');
- useEffect(() => {
- fetchGroupingLevels();
- return browserHistory.listen(handleRouteLeave);
- }, []);
- useEffect(() => {
- setSecondGrouping();
- }, [groupingLevels]);
- useEffect(() => {
- updateUrlWithNewLevel();
- }, [activeGroupingLevel]);
- useEffect(() => {
- fetchGroupingLevelDetails();
- }, [activeGroupingLevel, cursor]);
- function handleRouteLeave(newLocation: Location<{cursor?: string; level?: number}>) {
- if (
- newLocation.pathname === location.pathname ||
- (newLocation.pathname !== location.pathname &&
- newLocation.query.cursor === undefined &&
- newLocation.query.level === undefined)
- ) {
- return true;
- }
- // Removes cursor and level from the URL on route leave
- // so that the parameters will not interfere with other pages
- browserHistory.replace({
- pathname: newLocation.pathname,
- query: {
- ...newLocation.query,
- cursor: undefined,
- level: undefined,
- },
- });
- return false;
- }
- const handleSetActiveGroupingLevel = debounce((groupingLevelId: number | '') => {
- setActiveGroupingLevel(Number(groupingLevelId));
- }, DEFAULT_DEBOUNCE_DURATION);
- async function fetchGroupingLevels() {
- setIsLoading(true);
- setError(undefined);
- try {
- const response = await api.requestPromise(`/issues/${groupId}/grouping/levels/`);
- setIsLoading(false);
- setGroupingLevels(response.levels);
- } catch (err) {
- setIsLoading(false);
- setError(err);
- }
- }
- async function fetchGroupingLevelDetails() {
- if (!groupingLevels.length || !defined(activeGroupingLevel)) {
- return;
- }
- setIsGroupingLevelDetailsLoading(true);
- setError(undefined);
- try {
- const [data, , resp] = await api.requestPromise(
- `/issues/${groupId}/grouping/levels/${activeGroupingLevel}/new-issues/`,
- {
- method: 'GET',
- includeAllArgs: true,
- query: {
- ...location.query,
- per_page: 10,
- },
- }
- );
- const pageLinks = resp?.getResponseHeader?.('Link');
- setPagination(pageLinks ?? '');
- setActiveGroupingLevelDetails(Array.isArray(data) ? data : [data]);
- setIsGroupingLevelDetailsLoading(false);
- } catch (err) {
- setIsGroupingLevelDetailsLoading(false);
- setError(err);
- }
- }
- function updateUrlWithNewLevel() {
- if (!defined(activeGroupingLevel) || level === activeGroupingLevel) {
- return;
- }
- router.replace({
- pathname: location.pathname,
- query: {...location.query, cursor: undefined, level: activeGroupingLevel},
- });
- }
- function setSecondGrouping() {
- if (!groupingLevels.length) {
- return;
- }
- if (defined(level)) {
- if (!defined(groupingLevels[level])) {
- setError(t('The level you were looking for was not found.'));
- return;
- }
- if (level === activeGroupingLevel) {
- return;
- }
- setActiveGroupingLevel(level);
- return;
- }
- if (groupingLevels.length > 1) {
- setActiveGroupingLevel(groupingLevels[1].id);
- return;
- }
- setActiveGroupingLevel(groupingLevels[0].id);
- }
- if (isLoading) {
- return <LoadingIndicator />;
- }
- if (error) {
- return (
- <Fragment>
- <Layout.Body>
- <Layout.Main fullWidth>
- <ErrorWrapper>
- <ButtonBar gap={1}>
- <Button href={GROUPING_BREAKDOWN__DOC_LINK} external>
- {t('Read Docs')}
- </Button>
- <FeatureFeedback
- featureName="grouping"
- feedbackTypes={groupingFeedbackTypes}
- />
- </ButtonBar>
- <StyledErrorMessage
- onRetry={fetchGroupingLevels}
- groupId={groupId}
- error={error}
- projSlug={projSlug}
- orgSlug={organization.slug}
- hasProjectWriteAccess={organization.access.includes('project:write')}
- />
- </ErrorWrapper>
- </Layout.Main>
- </Layout.Body>
- </Fragment>
- );
- }
- if (!activeGroupingLevelDetails.length) {
- return <LoadingIndicator />;
- }
- const links = parseLinkHeader(pagination);
- const hasMore = links.previous?.results || links.next?.results;
- const paginationCurrentQuantity = activeGroupingLevelDetails.length;
- return (
- <Layout.Body>
- <Layout.Main fullWidth>
- <Wrapper>
- <Header>
- {t(
- 'This issue is an aggregate of multiple events that sentry determined originate from the same root-cause. Use this page to explore more detailed groupings that exist within this issue.'
- )}
- </Header>
- <Body>
- <Actions>
- <SliderWrapper>
- {t('Fewer issues')}
- <StyledRangeSlider
- name="grouping-level"
- allowedValues={groupingLevels.map(groupingLevel =>
- Number(groupingLevel.id)
- )}
- value={activeGroupingLevel ?? 0}
- onChange={handleSetActiveGroupingLevel}
- showLabel={false}
- />
- {t('More issues')}
- </SliderWrapper>
- <StyledButtonBar gap={1}>
- <Button href={GROUPING_BREAKDOWN__DOC_LINK} external>
- {t('Read Docs')}
- </Button>
- <FeatureFeedback
- featureName="grouping"
- feedbackTypes={groupingFeedbackTypes}
- />
- </StyledButtonBar>
- </Actions>
- <Content isReloading={isGroupingLevelDetailsLoading}>
- <StyledPanelTable headers={['', t('Events')]}>
- {activeGroupingLevelDetails.map(
- ({hash, title, metadata, latestEvent, eventCount}) => {
- // XXX(markus): Ugly hack to make NewIssue show the right things.
- return (
- <NewIssue
- key={hash}
- sampleEvent={{
- ...latestEvent,
- metadata: {
- ...(metadata || latestEvent.metadata),
- current_level: activeGroupingLevel,
- },
- title: title || latestEvent.title,
- }}
- eventCount={eventCount}
- organization={organization}
- />
- );
- }
- )}
- </StyledPanelTable>
- <StyledPagination
- pageLinks={pagination}
- disabled={isGroupingLevelDetailsLoading}
- caption={tct('Showing [current] of [total] [result]', {
- result: hasMore
- ? t('results')
- : tn('result', 'results', paginationCurrentQuantity),
- current: paginationCurrentQuantity,
- total: hasMore
- ? `${paginationCurrentQuantity}+`
- : paginationCurrentQuantity,
- })}
- />
- </Content>
- </Body>
- </Wrapper>
- </Layout.Main>
- </Layout.Body>
- );
- }
- export default withApi(Grouping);
- const Wrapper = styled('div')`
- flex: 1;
- display: grid;
- align-content: flex-start;
- margin: -${space(3)} -${space(4)};
- padding: ${space(3)} ${space(4)};
- `;
- const Header = styled('p')`
- && {
- margin-bottom: ${space(2)};
- }
- `;
- const Body = styled('div')`
- display: grid;
- gap: ${space(3)};
- `;
- const Actions = styled('div')`
- display: grid;
- align-items: center;
- gap: ${space(3)};
- @media (min-width: ${p => p.theme.breakpoints.small}) {
- grid-template-columns: 1fr max-content;
- gap: ${space(2)};
- }
- `;
- const StyledButtonBar = styled(ButtonBar)`
- justify-content: flex-start;
- `;
- const StyledErrorMessage = styled(ErrorMessage)`
- width: 100%;
- `;
- const ErrorWrapper = styled('div')`
- display: flex;
- flex-direction: column;
- align-items: flex-end;
- gap: ${space(1)};
- `;
- const StyledPanelTable = styled(PanelTable)`
- grid-template-columns: 1fr minmax(60px, auto);
- > * {
- padding: ${space(1.5)} ${space(2)};
- :nth-child(-n + 2) {
- padding: ${space(2)};
- }
- :nth-child(2n) {
- display: flex;
- text-align: right;
- justify-content: flex-end;
- }
- }
- @media (min-width: ${p => p.theme.breakpoints.xlarge}) {
- grid-template-columns: 1fr minmax(80px, auto);
- }
- `;
- const StyledPagination = styled(Pagination)`
- margin-top: 0;
- `;
- const Content = styled('div')<{isReloading: boolean}>`
- ${p =>
- p.isReloading &&
- `
- ${StyledPanelTable}, ${StyledPagination} {
- opacity: 0.5;
- pointer-events: none;
- }
- `}
- `;
- const SliderWrapper = styled('div')`
- display: grid;
- gap: ${space(1.5)};
- grid-template-columns: max-content max-content;
- justify-content: space-between;
- align-items: flex-start;
- position: relative;
- font-size: ${p => p.theme.fontSizeMedium};
- color: ${p => p.theme.subText};
- padding-bottom: ${space(2)};
- @media (min-width: 700px) {
- grid-template-columns: max-content minmax(270px, auto) max-content;
- align-items: center;
- justify-content: flex-start;
- padding-bottom: 0;
- }
- `;
- const StyledRangeSlider = styled(RangeSlider)`
- ${Slider} {
- background: transparent;
- margin-top: 0;
- margin-bottom: 0;
- ::-ms-thumb {
- box-shadow: 0 0 0 3px ${p => p.theme.backgroundSecondary};
- }
- ::-moz-range-thumb {
- box-shadow: 0 0 0 3px ${p => p.theme.backgroundSecondary};
- }
- ::-webkit-slider-thumb {
- box-shadow: 0 0 0 3px ${p => p.theme.backgroundSecondary};
- }
- }
- position: absolute;
- bottom: 0;
- left: ${space(1.5)};
- right: ${space(1.5)};
- @media (min-width: 700px) {
- position: static;
- left: auto;
- right: auto;
- }
- `;
|