import {Fragment} from 'react'; import styled from '@emotion/styled'; import * as Sentry from '@sentry/react'; import isEqual from 'lodash/isEqual'; import AsyncComponent from 'sentry/components/asyncComponent'; import RadioGroup from 'sentry/components/forms/controls/radioGroup'; import SelectControl from 'sentry/components/forms/controls/selectControl'; import Input from 'sentry/components/input'; import * as Layout from 'sentry/components/layouts/thirds'; import {t} from 'sentry/locale'; import {space} from 'sentry/styles/space'; import {Organization} from 'sentry/types'; import {IssueAlertRuleAction} from 'sentry/types/alerts'; import withOrganization from 'sentry/utils/withOrganization'; enum MetricValues { ERRORS, USERS, } enum Actions { ALERT_ON_EVERY_ISSUE, CUSTOMIZED_ALERTS, CREATE_ALERT_LATER, } const UNIQUE_USER_FREQUENCY_CONDITION = 'sentry.rules.conditions.event_frequency.EventUniqueUserFrequencyCondition'; const EVENT_FREQUENCY_CONDITION = 'sentry.rules.conditions.event_frequency.EventFrequencyCondition'; export const EVENT_FREQUENCY_PERCENT_CONDITION = 'sentry.rules.conditions.event_frequency.EventFrequencyPercentCondition'; export const REAPPEARED_EVENT_CONDITION = 'sentry.rules.conditions.reappeared_event.ReappearedEventCondition'; const ISSUE_ALERT_DEFAULT_ACTION: Omit< IssueAlertRuleAction, 'label' | 'name' | 'prompt' > = { id: 'sentry.mail.actions.NotifyEmailAction', targetType: 'IssueOwners', }; const METRIC_CONDITION_MAP = { [MetricValues.ERRORS]: EVENT_FREQUENCY_CONDITION, [MetricValues.USERS]: UNIQUE_USER_FREQUENCY_CONDITION, } as const; const DEFAULT_PLACEHOLDER_VALUE = '10'; type StateUpdater = (updatedData: RequestDataFragment) => void; type Props = AsyncComponent['props'] & { onChange: StateUpdater; organization: Organization; }; type State = AsyncComponent['state'] & { alertSetting: string; // TODO(ts): When we have alert conditional types, convert this conditions: any; interval: string; intervalChoices: [string, string][] | undefined; metric: MetricValues; threshold: string; }; type RequestDataFragment = { actionMatch: string; actions: Omit[]; conditions: {id: string; interval: string; value: string}[] | undefined; defaultRules: boolean; frequency: number; name: string; shouldCreateCustomRule: boolean; }; function getConditionFrom( interval: string, metricValue: MetricValues, threshold: string ): {id: string; interval: string; value: string} { let condition: string; switch (metricValue) { case MetricValues.ERRORS: condition = EVENT_FREQUENCY_CONDITION; break; case MetricValues.USERS: condition = UNIQUE_USER_FREQUENCY_CONDITION; break; default: throw new RangeError('Supplied metric value is not handled'); } return { interval, id: condition, value: threshold, }; } function unpackConditions(conditions: any[]) { const equalityReducer = (acc, curr) => { if (!acc || !curr || !isEqual(acc, curr)) { return null; } return acc; }; const intervalChoices = conditions .map(condition => condition.formFields?.interval?.choices) .reduce(equalityReducer); return {intervalChoices, interval: intervalChoices?.[0]?.[0]}; } class IssueAlertOptions extends AsyncComponent { getDefaultState(): State { return { ...super.getDefaultState(), conditions: [], intervalChoices: [], alertSetting: Actions.ALERT_ON_EVERY_ISSUE.toString(), metric: MetricValues.ERRORS, interval: '', threshold: '', }; } getAvailableMetricOptions() { return [ {value: MetricValues.ERRORS, label: t('occurrences of')}, {value: MetricValues.USERS, label: t('users affected by')}, ].filter(({value}) => { return this.state.conditions?.some?.( object => object?.id === METRIC_CONDITION_MAP[value] ); }); } getIssueAlertsChoices( hasProperlyLoadedConditions: boolean ): [string, string | React.ReactElement][] { const customizedAlertOption: [string, React.ReactNode] = [ Actions.CUSTOMIZED_ALERTS.toString(), { // XXX(epurkhiser): The `e.preventDefault` here is needed to stop // propagation of the click up to the label, causing it to focus // the radio input and lose focus on the select. e.preventDefault(); const alertSetting = Actions.CUSTOMIZED_ALERTS.toString(); this.setStateAndUpdateParents({alertSetting}); }} > {t('When there are more than')} this.setStateAndUpdateParents({threshold: threshold.target.value}) } data-test-id="range-input" /> this.setStateAndUpdateParents({metric: metric.value})} /> {t('a unique error in')} ({ value, label, }))} onChange={interval => this.setStateAndUpdateParents({interval: interval.value})} /> , ]; const options: [string, React.ReactNode][] = [ [Actions.ALERT_ON_EVERY_ISSUE.toString(), t('Alert me on every new issue')], ...(hasProperlyLoadedConditions ? [customizedAlertOption] : []), [Actions.CREATE_ALERT_LATER.toString(), t("I'll create my own alerts later")], ]; return options.map(([choiceValue, node]) => [ choiceValue, {node}, ]); } getUpdatedData(): RequestDataFragment { let defaultRules: boolean; let shouldCreateCustomRule: boolean; const alertSetting: Actions = parseInt(this.state.alertSetting, 10); switch (alertSetting) { case Actions.ALERT_ON_EVERY_ISSUE: defaultRules = true; shouldCreateCustomRule = false; break; case Actions.CREATE_ALERT_LATER: defaultRules = false; shouldCreateCustomRule = false; break; case Actions.CUSTOMIZED_ALERTS: defaultRules = false; shouldCreateCustomRule = true; break; default: throw new RangeError('Supplied alert creation action is not handled'); } return { defaultRules, shouldCreateCustomRule, name: 'Send a notification for new issues', conditions: this.state.interval.length > 0 && this.state.threshold.length > 0 ? [ getConditionFrom( this.state.interval, this.state.metric, this.state.threshold ), ] : undefined, actions: [ { ...ISSUE_ALERT_DEFAULT_ACTION, ...(this.props.organization.features.includes('issue-alert-fallback-targeting') ? {fallthroughType: 'ActiveMembers'} : {}), }, ], actionMatch: 'all', frequency: 5, }; } setStateAndUpdateParents( state: | (( prevState: Readonly, props: Readonly ) => Pick | State | null) | Pick | State | null ): void { this.setState(state, () => { this.props.onChange(this.getUpdatedData()); }); } getEndpoints(): ReturnType { return [['conditions', `/projects/${this.props.organization.slug}/rule-conditions/`]]; } onLoadAllEndpointsSuccess(): void { const conditions = this.state.conditions?.filter?.(object => Object.values(METRIC_CONDITION_MAP).includes(object?.id) ); if (!conditions || conditions.length === 0) { this.setStateAndUpdateParents({ conditions: undefined, }); return; } const {intervalChoices, interval} = unpackConditions(conditions); if (!intervalChoices || !interval) { Sentry.withScope(scope => { scope.setExtra('props', this.props); scope.setExtra('state', this.state); Sentry.captureException( new Error('Interval choices or sent from API endpoint is inconsistent or empty') ); }); this.setStateAndUpdateParents({ conditions: undefined, }); return; } this.setStateAndUpdateParents({ conditions, intervalChoices, interval, }); } renderBody(): React.ReactElement { const issueAlertOptionsChoices = this.getIssueAlertsChoices( this.state.conditions?.length > 0 ); return ( {t('2. Set your alert frequency')} this.setStateAndUpdateParents({alertSetting})} value={this.state.alertSetting} /> ); } } export default withOrganization(IssueAlertOptions); const Content = styled('div')` padding-top: ${space(2)}; padding-bottom: ${space(4)}; `; const CustomizeAlertsGrid = styled('div')` display: grid; grid-template-columns: repeat(5, max-content); gap: ${space(1)}; align-items: center; `; const InlineInput = styled(Input)` width: 80px; `; const InlineSelectControl = styled(SelectControl)` width: 160px; `; const RadioGroupWithPadding = styled(RadioGroup)` margin-bottom: ${space(2)}; `; const PageHeadingWithTopMargins = styled(Layout.Title)` margin-top: 65px; margin-bottom: 0; padding-bottom: ${space(3)}; border-bottom: 1px solid rgba(0, 0, 0, 0.1); `; const RadioItemWrapper = styled('div')` min-height: 35px; display: flex; flex-direction: column; justify-content: center; `;