123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302 |
- import {Fragment} from 'react';
- import {RouteComponentProps} from 'react-router';
- import styled from '@emotion/styled';
- import EmptyMessage from 'sentry/components/emptyMessage';
- import SelectField from 'sentry/components/forms/fields/selectField';
- import Form from 'sentry/components/forms/form';
- import JsonForm from 'sentry/components/forms/jsonForm';
- import Pagination from 'sentry/components/pagination';
- import Panel from 'sentry/components/panels/panel';
- import PanelBody from 'sentry/components/panels/panelBody';
- import PanelHeader from 'sentry/components/panels/panelHeader';
- import {fields} from 'sentry/data/forms/accountNotificationSettings';
- import {t} from 'sentry/locale';
- import {Organization, Project, UserEmail} from 'sentry/types';
- import withOrganizations from 'sentry/utils/withOrganizations';
- import DeprecatedAsyncView from 'sentry/views/deprecatedAsyncView';
- import {
- ACCOUNT_NOTIFICATION_FIELDS,
- FineTuneField,
- } from 'sentry/views/settings/account/notifications/fields';
- import NotificationSettingsByType, {
- OrganizationSelectHeader,
- } from 'sentry/views/settings/account/notifications/notificationSettingsByType';
- import {
- getNotificationTypeFromPathname,
- groupByOrganization,
- isGroupedByProject,
- } from 'sentry/views/settings/account/notifications/utils';
- import SettingsPageHeader from 'sentry/views/settings/components/settingsPageHeader';
- import TextBlock from 'sentry/views/settings/components/text/textBlock';
- const PanelBodyLineItem = styled(PanelBody)`
- font-size: 1rem;
- &:not(:last-child) {
- border-bottom: 1px solid ${p => p.theme.innerBorder};
- }
- `;
- const accountNotifications = [
- 'alerts',
- 'deploy',
- 'workflow',
- 'approval',
- 'quota',
- 'spikeProtection',
- ];
- type ANBPProps = {
- field: FineTuneField;
- projects: Project[];
- };
- function AccountNotificationsByProject({projects, field}: ANBPProps) {
- const projectsByOrg = groupByOrganization(projects);
- // eslint-disable-next-line @typescript-eslint/no-unused-vars
- const {title, description, ...fieldConfig} = field;
- // Display as select box in this view regardless of the type specified in the config
- const data = Object.values(projectsByOrg).map(org => ({
- name: org.organization.name,
- projects: org.projects.map(project => ({
- ...fieldConfig,
- // `name` key refers to field name
- // we use project.id because slugs are not unique across orgs
- name: project.id,
- label: project.slug,
- })),
- }));
- return (
- <Fragment>
- {data.map(({name, projects: projectFields}) => (
- <div key={name}>
- {projectFields.map(f => (
- <PanelBodyLineItem key={f.name}>
- <SelectField
- defaultValue={f.defaultValue}
- name={f.name}
- options={f.options}
- label={f.label}
- />
- </PanelBodyLineItem>
- ))}
- </div>
- ))}
- </Fragment>
- );
- }
- type ANBOProps = {
- field: FineTuneField;
- organizations: Organization[];
- };
- function AccountNotificationsByOrganization({organizations, field}: ANBOProps) {
- // eslint-disable-next-line @typescript-eslint/no-unused-vars
- const {title, description, ...fieldConfig} = field;
- // Display as select box in this view regardless of the type specified in the config
- const data = organizations.map(org => ({
- ...fieldConfig,
- // `name` key refers to field name
- // we use org.id to remain consistent project.id use (which is required because slugs are not unique across orgs)
- name: org.id,
- label: org.slug,
- }));
- return (
- <Fragment>
- {data.map(f => (
- <PanelBodyLineItem key={f.name}>
- <SelectField
- defaultValue={f.defaultValue}
- name={f.name}
- options={f.options}
- label={f.label}
- />
- </PanelBodyLineItem>
- ))}
- </Fragment>
- );
- }
- const AccountNotificationsByOrganizationContainer = withOrganizations(
- AccountNotificationsByOrganization
- );
- type Props = DeprecatedAsyncView['props'] &
- RouteComponentProps<{fineTuneType: string}, {}> & {
- organizations: Organization[];
- };
- type State = DeprecatedAsyncView['state'] & {
- emails: UserEmail[] | null;
- fineTuneData: Record<string, any> | null;
- notifications: Record<string, any> | null;
- organizationId: string | null;
- projects: Project[] | null;
- };
- class AccountNotificationFineTuning extends DeprecatedAsyncView<Props, State> {
- getDefaultState() {
- return {
- ...super.getDefaultState(),
- emails: [],
- fineTuneData: null,
- notifications: [],
- projects: [],
- organizationId: null,
- };
- }
- getEndpoints(): ReturnType<DeprecatedAsyncView['getEndpoints']> {
- const {fineTuneType: pathnameType} = this.props.params;
- const fineTuneType = getNotificationTypeFromPathname(pathnameType);
- const endpoints = [
- ['notifications', '/users/me/notifications/'],
- ['fineTuneData', `/users/me/notifications/${fineTuneType}/`],
- ];
- if (isGroupedByProject(fineTuneType) && this.props.organizations.length > 0) {
- const orgId = this.state?.organizationId || this.props.organizations[0].id;
- endpoints.push(['projects', `/projects/?organization_id=${orgId}`]);
- }
- endpoints.push(['emails', '/users/me/emails/']);
- if (fineTuneType === 'email') {
- endpoints.push(['emails', '/users/me/emails/']);
- }
- return endpoints as ReturnType<DeprecatedAsyncView['getEndpoints']>;
- }
- // Return a sorted list of user's verified emails
- get emailChoices() {
- return (
- this.state.emails
- ?.filter(({isVerified}) => isVerified)
- ?.sort((a, b) => {
- // Sort by primary -> email
- if (a.isPrimary) {
- return -1;
- }
- if (b.isPrimary) {
- return 1;
- }
- return a.email < b.email ? -1 : 1;
- }) ?? []
- );
- }
- handleOrgChange = (option: {label: string; value: string}) => {
- this.setState({organizationId: option.value});
- const self = this;
- setTimeout(() => {
- self.reloadData();
- }, 0);
- };
- renderBody() {
- const {params} = this.props;
- const {fineTuneType: pathnameType} = params;
- const fineTuneType = getNotificationTypeFromPathname(pathnameType);
- if (accountNotifications.includes(fineTuneType)) {
- return <NotificationSettingsByType notificationType={fineTuneType} />;
- }
- const {notifications, projects, fineTuneData, projectsPageLinks} = this.state;
- const isProject =
- isGroupedByProject(fineTuneType) && this.props.organizations.length > 0;
- const field = ACCOUNT_NOTIFICATION_FIELDS[fineTuneType];
- const {title, description} = field;
- const [stateKey, url] = isProject ? this.getEndpoints()[2] : [];
- const hasProjects = !!projects?.length;
- if (fineTuneType === 'email') {
- // Fetch verified email addresses
- field.options = this.emailChoices.map(({email}) => ({value: email, label: email}));
- }
- if (!notifications || !fineTuneData) {
- return null;
- }
- return (
- <div>
- <SettingsPageHeader title={title} />
- {description && <TextBlock>{description}</TextBlock>}
- {field &&
- field.defaultFieldName &&
- // not implemented yet
- field.defaultFieldName !== 'weeklyReports' && (
- <Form
- saveOnBlur
- apiMethod="PUT"
- apiEndpoint="/users/me/notifications/"
- initialData={notifications}
- >
- <JsonForm
- title={`Default ${title}`}
- fields={[fields[field.defaultFieldName]]}
- />
- </Form>
- )}
- <Panel>
- <PanelHeader hasButtons={isProject}>
- {isProject ? (
- <Fragment>
- <OrganizationSelectHeader
- organizations={this.props.organizations}
- organizationId={this.state.organizationId || ''}
- handleOrgChange={this.handleOrgChange}
- />
- {this.renderSearchInput({
- placeholder: t('Search Projects'),
- url,
- stateKey,
- })}
- </Fragment>
- ) : (
- <Heading>{t('Organizations')}</Heading>
- )}
- </PanelHeader>
- <PanelBody>
- <Form
- saveOnBlur
- apiMethod="PUT"
- apiEndpoint={`/users/me/notifications/${fineTuneType}/`}
- initialData={fineTuneData}
- >
- {isProject && hasProjects && (
- <AccountNotificationsByProject projects={projects!} field={field} />
- )}
- {isProject && !hasProjects && (
- <EmptyMessage>{t('No projects found')}</EmptyMessage>
- )}
- {!isProject && (
- <AccountNotificationsByOrganizationContainer field={field} />
- )}
- </Form>
- </PanelBody>
- </Panel>
- {projects && <Pagination pageLinks={projectsPageLinks} {...this.props} />}
- </div>
- );
- }
- }
- const Heading = styled('div')`
- flex: 1;
- `;
- export default withOrganizations(AccountNotificationFineTuning);
|