123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107 |
- import type {RouteComponentProps} from 'react-router';
- import {hasEveryAccess} from 'sentry/components/acl/access';
- import Form from 'sentry/components/forms/form';
- import JsonForm from 'sentry/components/forms/jsonForm';
- import ExternalLink from 'sentry/components/links/externalLink';
- import LoadingError from 'sentry/components/loadingError';
- import LoadingIndicator from 'sentry/components/loadingIndicator';
- import SentryDocumentTitle from 'sentry/components/sentryDocumentTitle';
- import {fields} from 'sentry/data/forms/projectIssueGrouping';
- import {t, tct} from 'sentry/locale';
- import ProjectsStore from 'sentry/stores/projectsStore';
- import type {EventGroupingConfig} from 'sentry/types/event';
- import type {Organization} from 'sentry/types/organization';
- import type {Project} from 'sentry/types/project';
- import {useApiQuery} from 'sentry/utils/queryClient';
- import routeTitleGen from 'sentry/utils/routeTitle';
- import SettingsPageHeader from 'sentry/views/settings/components/settingsPageHeader';
- import TextBlock from 'sentry/views/settings/components/text/textBlock';
- import PermissionAlert from 'sentry/views/settings/project/permissionAlert';
- type Props = RouteComponentProps<{}, {projectId: string}> & {
- organization: Organization;
- project: Project;
- };
- export default function ProjectIssueGrouping({organization, project, params}: Props) {
- const queryKey = `/projects/${organization.slug}/${project.slug}/grouping-configs/`;
- const {
- data: groupingConfigs,
- isLoading,
- isError,
- refetch,
- } = useApiQuery<EventGroupingConfig[]>([queryKey], {staleTime: 0, cacheTime: 0});
- if (isLoading) {
- return <LoadingIndicator />;
- }
- if (isError) {
- return (
- <LoadingError message={t('Failed to load grouping configs')} onRetry={refetch} />
- );
- }
- const handleSubmit = (response: Project) => {
- // This will update our project context
- ProjectsStore.onUpdateSuccess(response);
- };
- const endpoint = `/projects/${organization.slug}/${project.slug}/`;
- const access = new Set(organization.access.concat(project.access));
- const hasAccess = hasEveryAccess(['project:write'], {organization, project});
- const jsonFormProps = {
- additionalFieldProps: {
- organization,
- groupingConfigs,
- },
- features: new Set(organization.features),
- access,
- disabled: !hasAccess,
- };
- return (
- <SentryDocumentTitle
- title={routeTitleGen(t('Issue Grouping'), params.projectId, false)}
- >
- <SettingsPageHeader title={t('Issue Grouping')} />
- <TextBlock>
- {tct(
- `All events have a fingerprint. Events with the same fingerprint are grouped together into an issue. To learn more about issue grouping, [link: read the docs].`,
- {
- link: (
- <ExternalLink href="https://docs.sentry.io/product/data-management-settings/event-grouping/" />
- ),
- }
- )}
- </TextBlock>
- <PermissionAlert project={project} />
- <Form
- saveOnBlur
- allowUndo
- initialData={project}
- apiMethod="PUT"
- apiEndpoint={endpoint}
- onSubmitSuccess={handleSubmit}
- >
- <JsonForm
- {...jsonFormProps}
- title={t('Fingerprint Rules')}
- fields={[fields.fingerprintingRules]}
- />
- <JsonForm
- {...jsonFormProps}
- title={t('Stack Trace Rules')}
- fields={[fields.groupingEnhancements]}
- />
- </Form>
- </SentryDocumentTitle>
- );
- }
|