123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319 |
- import {Fragment} from 'react';
- import {RouteComponentProps} from 'react-router';
- import {openEditOwnershipRules, openModal} from 'sentry/actionCreators/modal';
- import Access, {hasEveryAccess} from 'sentry/components/acl/access';
- import {Button} from 'sentry/components/button';
- import ButtonBar from 'sentry/components/buttonBar';
- import ErrorBoundary from 'sentry/components/errorBoundary';
- import Form from 'sentry/components/forms/form';
- import JsonForm from 'sentry/components/forms/jsonForm';
- import ExternalLink from 'sentry/components/links/externalLink';
- import {IconEdit} from 'sentry/icons';
- import {t, tct} from 'sentry/locale';
- import {CodeOwner, IssueOwnership, Organization, Project} from 'sentry/types';
- import routeTitleGen from 'sentry/utils/routeTitle';
- import AsyncView from 'sentry/views/asyncView';
- 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';
- import AddCodeOwnerModal from 'sentry/views/settings/project/projectOwnership/addCodeOwnerModal';
- import {CodeOwnerErrors} from 'sentry/views/settings/project/projectOwnership/codeownerErrors';
- import {CodeOwnerFileTable} from 'sentry/views/settings/project/projectOwnership/codeOwnerFileTable';
- import CodeOwnersPanel from 'sentry/views/settings/project/projectOwnership/codeowners';
- import {OwnershipRulesTable} from 'sentry/views/settings/project/projectOwnership/ownershipRulesTable';
- import RulesPanel from 'sentry/views/settings/project/projectOwnership/rulesPanel';
- type Props = {
- organization: Organization;
- project: Project;
- } & RouteComponentProps<{projectId: string}, {}>;
- type State = {
- codeowners?: CodeOwner[];
- ownership?: null | IssueOwnership;
- } & AsyncView['state'];
- class ProjectOwnership extends AsyncView<Props, State> {
- // TODO: Remove with `streamline-targeting-context`
- getOwnershipTitle() {
- const {organization} = this.props;
- return organization.features?.includes('streamline-targeting-context')
- ? t('Ownership Rules')
- : t('Issue Owners');
- }
- getTitle() {
- const {project} = this.props;
- return routeTitleGen(this.getOwnershipTitle(), project.slug, false);
- }
- getEndpoints(): ReturnType<AsyncView['getEndpoints']> {
- const {organization, project} = this.props;
- const endpoints: ReturnType<AsyncView['getEndpoints']> = [
- ['ownership', `/projects/${organization.slug}/${project.slug}/ownership/`],
- ];
- if (organization.features.includes('integrations-codeowners')) {
- endpoints.push([
- 'codeowners',
- `/projects/${organization.slug}/${project.slug}/codeowners/`,
- {query: {expand: ['codeMapping', 'ownershipSyntax']}},
- ]);
- }
- return endpoints;
- }
- handleAddCodeOwner = () => {
- openModal(modalProps => (
- <AddCodeOwnerModal
- {...modalProps}
- organization={this.props.organization}
- project={this.props.project}
- onSave={this.handleCodeOwnerAdded}
- />
- ));
- };
- getPlaceholder() {
- return `#example usage
- path:src/example/pipeline/* person@sentry.io #infra
- module:com.module.name.example #sdks
- url:http://example.com/settings/* #product
- tags.sku_class:enterprise #enterprise`;
- }
- handleOwnershipSave = (text: string | null) => {
- this.setState(prevState => ({
- ...(prevState.ownership
- ? {
- ownership: {
- ...prevState.ownership,
- raw: text || '',
- },
- }
- : {}),
- }));
- };
- handleCodeOwnerAdded = (data: CodeOwner) => {
- const {codeowners} = this.state;
- const newCodeowners = [data, ...(codeowners || [])];
- this.setState({codeowners: newCodeowners});
- };
- handleCodeOwnerDeleted = (data: CodeOwner) => {
- const {codeowners} = this.state;
- const newCodeowners = (codeowners || []).filter(
- codeowner => codeowner.id !== data.id
- );
- this.setState({codeowners: newCodeowners});
- };
- handleCodeOwnerUpdated = (data: CodeOwner) => {
- const codeowners = this.state.codeowners || [];
- const index = codeowners.findIndex(item => item.id === data.id);
- this.setState({
- codeowners: [...codeowners.slice(0, index), data, ...codeowners.slice(index + 1)],
- });
- };
- renderBody() {
- const {project, organization} = this.props;
- const {ownership, codeowners} = this.state;
- const disabled = !hasEveryAccess(['project:write'], {organization, project});
- const editOwnershipRulesDisabled = !hasEveryAccess(['project:read'], {
- organization,
- project,
- });
- const hasStreamlineTargetingContext = organization.features?.includes(
- 'streamline-targeting-context'
- );
- const hasCodeowners = organization.features?.includes('integrations-codeowners');
- return (
- <Fragment>
- <SettingsPageHeader
- title={this.getOwnershipTitle()}
- action={
- <ButtonBar gap={1}>
- {hasCodeowners && (
- <Access access={['org:integrations']} project={project}>
- {({hasAccess}) => (
- <Button
- onClick={this.handleAddCodeOwner}
- size="sm"
- data-test-id="add-codeowner-button"
- disabled={!hasAccess}
- >
- {t('Import CODEOWNERS')}
- </Button>
- )}
- </Access>
- )}
- {hasStreamlineTargetingContext && (
- <Button
- type="button"
- size="sm"
- icon={<IconEdit size="xs" />}
- priority="primary"
- onClick={() =>
- openEditOwnershipRules({
- organization,
- project,
- ownership: ownership!,
- onSave: this.handleOwnershipSave,
- })
- }
- disabled={!!ownership && editOwnershipRulesDisabled}
- >
- {t('Edit Rules')}
- </Button>
- )}
- </ButtonBar>
- }
- />
- <TextBlock>
- {tct(
- `Auto-assign issues to users and teams. To learn more, [link:read the docs].`,
- {
- link: (
- <ExternalLink href="https://docs.sentry.io/product/error-monitoring/issue-owners/" />
- ),
- }
- )}
- </TextBlock>
- <PermissionAlert
- access={!editOwnershipRulesDisabled ? ['project:read'] : ['project:write']}
- project={project}
- />
- <CodeOwnerErrors
- orgSlug={organization.slug}
- projectSlug={project.slug}
- codeowners={codeowners ?? []}
- />
- {hasStreamlineTargetingContext && ownership && (
- <ErrorBoundary mini>
- <OwnershipRulesTable
- projectRules={ownership.schema?.rules ?? []}
- codeowners={codeowners ?? []}
- />
- </ErrorBoundary>
- )}
- {!hasStreamlineTargetingContext && ownership && (
- <RulesPanel
- data-test-id="issueowners-panel"
- type="issueowners"
- raw={ownership.raw || ''}
- dateUpdated={ownership.lastUpdated}
- placeholder={this.getPlaceholder()}
- controls={[
- <Button
- key="edit"
- size="xs"
- onClick={() =>
- openEditOwnershipRules({
- organization,
- project,
- ownership,
- onSave: this.handleOwnershipSave,
- })
- }
- disabled={editOwnershipRulesDisabled}
- >
- {t('Edit')}
- </Button>,
- ]}
- />
- )}
- <PermissionAlert project={project} />
- {hasCodeowners &&
- (hasStreamlineTargetingContext ? (
- <CodeOwnerFileTable
- project={project}
- codeowners={codeowners ?? []}
- onDelete={this.handleCodeOwnerDeleted}
- onUpdate={this.handleCodeOwnerUpdated}
- disabled={disabled}
- />
- ) : (
- <CodeOwnersPanel
- codeowners={codeowners || []}
- onDelete={this.handleCodeOwnerDeleted}
- onUpdate={this.handleCodeOwnerUpdated}
- disabled={disabled}
- {...this.props}
- />
- ))}
- {ownership && (
- <Form
- apiEndpoint={`/projects/${organization.slug}/${project.slug}/ownership/`}
- apiMethod="PUT"
- saveOnBlur
- initialData={{
- fallthrough: ownership.fallthrough,
- autoAssignment: ownership.autoAssignment,
- codeownersAutoSync: ownership.codeownersAutoSync,
- }}
- hideFooter
- >
- <JsonForm
- forms={[
- {
- title: t('Issue Owners'),
- fields: [
- {
- name: 'autoAssignment',
- type: 'choice',
- label: t('Prioritize Auto Assignment'),
- help: t(
- "When there's a conflict between suspect commit and ownership rules."
- ),
- choices: [
- [
- 'Auto Assign to Suspect Commits',
- t('Auto-assign to suspect commits'),
- ],
- ['Auto Assign to Issue Owner', t('Auto-assign to issue owner')],
- ['Turn off Auto-Assignment', t('Turn off auto-assignment')],
- ],
- disabled,
- },
- ...(organization.features.includes('issue-alert-fallback-targeting')
- ? []
- : [
- {
- name: 'fallthrough',
- type: 'boolean' as const,
- label: t(
- 'Send alert to project members if there’s no assigned owner'
- ),
- help: t(
- 'Alerts will be sent to all users who have access to this project.'
- ),
- disabled,
- },
- ]),
- {
- name: 'codeownersAutoSync',
- type: 'boolean',
- label: t('Sync changes from CODEOWNERS'),
- help: t(
- 'We’ll update any changes you make to your CODEOWNERS files during a release.'
- ),
- disabled: disabled || !(this.state.codeowners || []).length,
- },
- ],
- },
- ]}
- />
- </Form>
- )}
- </Fragment>
- );
- }
- }
- export default ProjectOwnership;
|