import {ChangeEvent, Fragment, ReactNode} from 'react'; import {browserHistory, RouteComponentProps} from 'react-router'; import {components} from 'react-select'; import styled from '@emotion/styled'; import classNames from 'classnames'; import {Location} from 'history'; import cloneDeep from 'lodash/cloneDeep'; import debounce from 'lodash/debounce'; import omit from 'lodash/omit'; import set from 'lodash/set'; import { addErrorMessage, addLoadingMessage, addSuccessMessage, } from 'sentry/actionCreators/indicator'; import {updateOnboardingTask} from 'sentry/actionCreators/onboardingTasks'; import {hasEveryAccess} from 'sentry/components/acl/access'; import {Alert} from 'sentry/components/alert'; import {Button} from 'sentry/components/button'; import Confirm from 'sentry/components/confirm'; import SelectControl from 'sentry/components/forms/controls/selectControl'; import FieldGroup from 'sentry/components/forms/fieldGroup'; import FieldHelp from 'sentry/components/forms/fieldGroup/fieldHelp'; import SelectField from 'sentry/components/forms/fields/selectField'; import Form, {FormProps} from 'sentry/components/forms/form'; import FormField from 'sentry/components/forms/formField'; import IdBadge from 'sentry/components/idBadge'; import Input from 'sentry/components/input'; import * as Layout from 'sentry/components/layouts/thirds'; import ExternalLink from 'sentry/components/links/externalLink'; import List from 'sentry/components/list'; import ListItem from 'sentry/components/list/listItem'; import LoadingMask from 'sentry/components/loadingMask'; import {CursorHandler} from 'sentry/components/pagination'; import Panel from 'sentry/components/panels/panel'; import PanelBody from 'sentry/components/panels/panelBody'; import TeamSelector from 'sentry/components/teamSelector'; import {Tooltip} from 'sentry/components/tooltip'; import {ALL_ENVIRONMENTS_KEY} from 'sentry/constants'; import {IconChevron} from 'sentry/icons'; import {t, tct, tn} from 'sentry/locale'; import GroupStore from 'sentry/stores/groupStore'; import {space} from 'sentry/styles/space'; import { Environment, IssueOwnership, Member, OnboardingTaskKey, Organization, Project, Team, } from 'sentry/types'; import { IssueAlertRule, IssueAlertRuleAction, IssueAlertRuleActionTemplate, IssueAlertRuleConditionTemplate, UnsavedIssueAlertRule, } from 'sentry/types/alerts'; import {metric, trackAnalytics} from 'sentry/utils/analytics'; import {getDisplayName} from 'sentry/utils/environment'; import {isActiveSuperuser} from 'sentry/utils/isActiveSuperuser'; import recreateRoute from 'sentry/utils/recreateRoute'; import routeTitleGen from 'sentry/utils/routeTitle'; import {normalizeUrl} from 'sentry/utils/withDomainRequired'; import withOrganization from 'sentry/utils/withOrganization'; import withProjects from 'sentry/utils/withProjects'; import PreviewTable from 'sentry/views/alerts/rules/issue/previewTable'; import { CHANGE_ALERT_CONDITION_IDS, CHANGE_ALERT_PLACEHOLDERS_LABELS, } from 'sentry/views/alerts/utils/constants'; import DeprecatedAsyncView from 'sentry/views/deprecatedAsyncView'; import PermissionAlert from 'sentry/views/settings/project/permissionAlert'; import {getProjectOptions} from '../utils'; import RuleNodeList from './ruleNodeList'; import SetupAlertIntegrationButton from './setupAlertIntegrationButton'; const FREQUENCY_OPTIONS = [ {value: '5', label: t('5 minutes')}, {value: '10', label: t('10 minutes')}, {value: '30', label: t('30 minutes')}, {value: '60', label: t('60 minutes')}, {value: '180', label: t('3 hours')}, {value: '720', label: t('12 hours')}, {value: '1440', label: t('24 hours')}, {value: '10080', label: t('1 week')}, {value: '43200', label: t('30 days')}, ]; const ACTION_MATCH_OPTIONS = [ {value: 'all', label: t('all')}, {value: 'any', label: t('any')}, {value: 'none', label: t('none')}, ]; const ACTION_MATCH_OPTIONS_MIGRATED = [ {value: 'all', label: t('all')}, {value: 'any', label: t('any')}, ]; const defaultRule: UnsavedIssueAlertRule = { actionMatch: 'any', filterMatch: 'all', actions: [], // note we update the default conditions in onLoadAllEndpointsSuccess conditions: [], filters: [], name: '', frequency: 60 * 24, environment: ALL_ENVIRONMENTS_KEY, }; const POLLING_MAX_TIME_LIMIT = 3 * 60000; const SENTRY_ISSUE_ALERT_DOCS_URL = 'https://docs.sentry.io/product/alerts/alert-types/#issue-alerts'; type ConditionOrActionProperty = 'conditions' | 'actions' | 'filters'; type RuleTaskResponse = { status: 'pending' | 'failed' | 'success'; error?: string; rule?: IssueAlertRule; }; type RouteParams = {projectId?: string; ruleId?: string}; export type IncompatibleRule = { conditionIndices: number[] | null; filterIndices: number[] | null; }; type Props = { location: Location; members: Member[] | undefined; organization: Organization; project: Project; projects: Project[]; userTeamIds: string[]; loadingProjects?: boolean; onChangeTitle?: (data: string) => void; } & RouteComponentProps; type State = DeprecatedAsyncView['state'] & { configs: { actions: IssueAlertRuleActionTemplate[]; conditions: IssueAlertRuleConditionTemplate[]; filters: IssueAlertRuleConditionTemplate[]; } | null; detailedError: null | { [key: string]: string[]; }; environments: Environment[] | null; incompatibleConditions: number[] | null; incompatibleFilters: number[] | null; issueCount: number; loadingPreview: boolean; previewCursor: string | null | undefined; previewEndpoint: null | string; previewError: null | string; previewGroups: string[] | null; previewPage: number; project: Project; sendingNotification: boolean; uuid: null | string; duplicateTargetRule?: UnsavedIssueAlertRule | IssueAlertRule | null; ownership?: null | IssueOwnership; rule?: UnsavedIssueAlertRule | IssueAlertRule | null; }; function isSavedAlertRule(rule: State['rule']): rule is IssueAlertRule { return rule?.hasOwnProperty('id') ?? false; } class IssueRuleEditor extends DeprecatedAsyncView { pollingTimeout: number | undefined = undefined; trackIncompatibleAnalytics: boolean = false; isUnmounted = false; get isDuplicateRule(): boolean { const {location} = this.props; const createFromDuplicate = location?.query.createFromDuplicate === 'true'; return createFromDuplicate && location?.query.duplicateRuleId; } componentDidMount() { super.componentDidMount(); this.fetchPreview(); } componentWillUnmount() { super.componentWillUnmount(); this.isUnmounted = true; GroupStore.reset(); window.clearTimeout(this.pollingTimeout); this.checkIncompatibleRuleDebounced.cancel(); this.fetchPreviewDebounced.cancel(); } componentDidUpdate(_prevProps: Props, prevState: State) { if (prevState.previewCursor !== this.state.previewCursor) { this.fetchPreview(); } else if (this.isRuleStateChange(prevState)) { this.setState({ loadingPreview: true, incompatibleConditions: null, incompatibleFilters: null, }); this.fetchPreviewDebounced(); this.checkIncompatibleRuleDebounced(); } if (prevState.project.id === this.state.project.id) { return; } this.fetchEnvironments(); } isRuleStateChange(prevState: State): boolean { const prevRule = prevState.rule; const curRule = this.state.rule; return ( JSON.stringify(prevRule?.conditions) !== JSON.stringify(curRule?.conditions) || JSON.stringify(prevRule?.filters) !== JSON.stringify(curRule?.filters) || prevRule?.actionMatch !== curRule?.actionMatch || prevRule?.filterMatch !== curRule?.filterMatch || prevRule?.frequency !== curRule?.frequency || JSON.stringify(prevState.project) !== JSON.stringify(this.state.project) ); } getTitle() { const {organization} = this.props; const {rule, project} = this.state; const ruleName = rule?.name; return routeTitleGen( ruleName ? t('Alert - %s', ruleName) : t('New Alert Rule'), organization.slug, false, project?.slug ); } getDefaultState() { const {userTeamIds, project} = this.props; const defaultState = { ...super.getDefaultState(), configs: null, detailedError: null, rule: {...defaultRule}, environments: [], uuid: null, project, previewGroups: null, previewCursor: null, previewError: null, issueCount: 0, previewPage: 0, loadingPreview: false, sendingNotification: false, incompatibleConditions: null, incompatibleFilters: null, previewEndpoint: null, }; const projectTeamIds = new Set(project.teams.map(({id}) => id)); const userTeamId = userTeamIds.find(id => projectTeamIds.has(id)) ?? null; defaultState.rule.owner = userTeamId && `team:${userTeamId}`; return defaultState; } getEndpoints(): ReturnType { const { location: {query}, params: {ruleId}, } = this.props; const {organization} = this.props; // project in state isn't initialized when getEndpoints is first called const project = this.state?.project ?? this.props.project; const endpoints = [ [ 'environments', `/projects/${organization.slug}/${project.slug}/environments/`, { query: { visibility: 'visible', }, }, ], ['configs', `/projects/${organization.slug}/${project.slug}/rules/configuration/`], ['ownership', `/projects/${organization.slug}/${project.slug}/ownership/`], ]; if (ruleId) { endpoints.push([ 'rule', `/projects/${organization.slug}/${project.slug}/rules/${ruleId}/`, ]); } if (!ruleId && query.createFromDuplicate && query.duplicateRuleId) { endpoints.push([ 'duplicateTargetRule', `/projects/${organization.slug}/${project.slug}/rules/${query.duplicateRuleId}/`, ]); } return endpoints as [string, string][]; } onRequestSuccess({stateKey, data}) { if (stateKey === 'rule' && data.name) { this.props.onChangeTitle?.(data.name); } if (stateKey === 'duplicateTargetRule') { this.setState({ rule: { ...omit(data, ['id']), name: data.name + ' copy', } as UnsavedIssueAlertRule, }); } } onLoadAllEndpointsSuccess() { const {rule} = this.state; const { params: {ruleId}, } = this.props; if (rule) { ((rule as IssueAlertRule)?.errors || []).map(({detail}) => addErrorMessage(detail, {append: true}) ); } if (!ruleId) { // now that we've loaded all the possible conditions, we can populate the // value of conditions for a new alert const id = 'sentry.rules.conditions.first_seen_event.FirstSeenEventCondition'; this.handleChange('conditions', [ { id, label: CHANGE_ALERT_PLACEHOLDERS_LABELS[id], name: 'A new issue is created', }, ]); } } pollHandler = async (quitTime: number) => { if (Date.now() > quitTime) { addErrorMessage(t('Looking for that channel took too long :(')); this.setState({loading: false}); return; } const {organization} = this.props; const {uuid, project} = this.state; const origRule = this.state.rule; try { const response: RuleTaskResponse = await this.api.requestPromise( `/projects/${organization.slug}/${project.slug}/rule-task/${uuid}/` ); const {status, rule, error} = response; if (status === 'pending') { window.clearTimeout(this.pollingTimeout); this.pollingTimeout = window.setTimeout(() => { this.pollHandler(quitTime); }, 1000); return; } if (status === 'failed') { this.setState({ detailedError: {actions: [error ? error : t('An error occurred')]}, loading: false, }); this.handleRuleSaveFailure(t('An error occurred')); } if (rule) { const ruleId = isSavedAlertRule(origRule) ? `${origRule.id}/` : ''; const isNew = !ruleId; this.handleRuleSuccess(isNew, rule); } } catch { this.handleRuleSaveFailure(t('An error occurred')); this.setState({loading: false}); } }; fetchPreview = (resetCursor = false) => { const {organization} = this.props; const {project, rule, previewCursor, previewEndpoint} = this.state; if (!rule) { return; } this.setState({loadingPreview: true}); if (resetCursor) { this.setState({previewCursor: null, previewPage: 0}); } // we currently don't have a way to parse objects from query params, so this method is POST for now this.api .requestPromise(`/projects/${organization.slug}/${project.slug}/rules/preview/`, { method: 'POST', includeAllArgs: true, query: { cursor: resetCursor ? null : previewCursor, per_page: 5, }, data: { conditions: rule?.conditions || [], filters: rule?.filters || [], actionMatch: rule?.actionMatch || 'all', filterMatch: rule?.filterMatch || 'all', frequency: rule?.frequency || 60, endpoint: previewEndpoint, }, }) .then(([data, _, resp]) => { if (this.isUnmounted) { return; } GroupStore.add(data); const pageLinks = resp?.getResponseHeader('Link'); const hits = resp?.getResponseHeader('X-Hits'); const endpoint = resp?.getResponseHeader('Endpoint'); const issueCount = typeof hits !== 'undefined' && hits ? parseInt(hits, 10) || 0 : 0; this.setState({ previewGroups: data.map(g => g.id), previewError: null, pageLinks: pageLinks ?? '', issueCount, loadingPreview: false, previewEndpoint: endpoint ?? null, }); }) .catch(_ => { const errorMessage = rule?.conditions.length || rule?.filters.length ? t('Preview is not supported for these conditions') : t('Select a condition to generate a preview'); this.setState({ previewError: errorMessage, loadingPreview: false, }); }); }; fetchPreviewDebounced = debounce(() => { this.fetchPreview(true); }, 1000); // As more incompatible combinations are added, we will need a more generic way to check for incompatibility. checkIncompatibleRuleDebounced = debounce(() => { const {conditionIndices, filterIndices} = findIncompatibleRules(this.state.rule); if ( !this.trackIncompatibleAnalytics && (conditionIndices !== null || filterIndices !== null) ) { this.trackIncompatibleAnalytics = true; trackAnalytics('edit_alert_rule.incompatible_rule', { organization: this.props.organization, }); } this.setState({ incompatibleConditions: conditionIndices, incompatibleFilters: filterIndices, }); }, 500); onPreviewCursor: CursorHandler = (cursor, _1, _2, direction) => { this.setState({ previewCursor: cursor, previewPage: this.state.previewPage + direction, }); }; fetchEnvironments() { const {organization} = this.props; const {project} = this.state; this.api .requestPromise(`/projects/${organization.slug}/${project.slug}/environments/`, { query: { visibility: 'visible', }, }) .then(response => this.setState({environments: response})) .catch(_err => addErrorMessage(t('Unable to fetch environments'))); } fetchStatus() { // pollHandler calls itself until it gets either a success // or failed status but we don't want to poll forever so we pass // in a hard stop time of 3 minutes before we bail. const quitTime = Date.now() + POLLING_MAX_TIME_LIMIT; window.clearTimeout(this.pollingTimeout); this.pollingTimeout = window.setTimeout(() => { this.pollHandler(quitTime); }, 1000); } testNotifications = () => { const {organization} = this.props; const {project, rule} = this.state; this.setState({sendingNotification: true}); const actions = rule?.actions ? rule?.actions.length : 0; addLoadingMessage( tn('Sending a test notification...', 'Sending test notifications...', actions) ); this.api .requestPromise(`/projects/${organization.slug}/${project.slug}/rule-actions/`, { method: 'POST', data: { actions: rule?.actions ?? [], }, }) .then(() => { addSuccessMessage(tn('Notification sent!', 'Notifications sent!', actions)); trackAnalytics('edit_alert_rule.notification_test', { organization, success: true, }); }) .catch(() => { addErrorMessage(tn('Notification failed', 'Notifications failed', actions)); trackAnalytics('edit_alert_rule.notification_test', { organization, success: false, }); }) .finally(() => { this.setState({sendingNotification: false}); }); }; handleRuleSuccess = (isNew: boolean, rule: IssueAlertRule) => { const {organization, router} = this.props; const {project} = this.state; this.setState({detailedError: null, loading: false, rule}); // The onboarding task will be completed on the server side when the alert // is created updateOnboardingTask(null, organization, { task: OnboardingTaskKey.ALERT_RULE, status: 'complete', }); metric.endTransaction({name: 'saveAlertRule'}); router.push( normalizeUrl({ pathname: `/organizations/${organization.slug}/alerts/rules/${project.slug}/${rule.id}/details/`, }) ); addSuccessMessage(isNew ? t('Created alert rule') : t('Updated alert rule')); }; handleRuleSaveFailure(msg: ReactNode) { addErrorMessage(msg); metric.endTransaction({name: 'saveAlertRule'}); } handleSubmit = async () => { const {project, rule} = this.state; const ruleId = isSavedAlertRule(rule) ? `${rule.id}/` : ''; const isNew = !ruleId; const {organization} = this.props; const endpoint = `/projects/${organization.slug}/${project.slug}/rules/${ruleId}`; if (rule && rule.environment === ALL_ENVIRONMENTS_KEY) { delete rule.environment; } addLoadingMessage(); try { const transaction = metric.startTransaction({name: 'saveAlertRule'}); transaction.setTag('type', 'issue'); transaction.setTag('operation', isNew ? 'create' : 'edit'); if (rule) { for (const action of rule.actions) { // Grab the last part of something like 'sentry.mail.actions.NotifyEmailAction' const splitActionId = action.id.split('.'); const actionName = splitActionId[splitActionId.length - 1]; if (actionName === 'SlackNotifyServiceAction') { transaction.setTag(actionName, true); } } transaction.setData('actions', rule.actions); } const [data, , resp] = await this.api.requestPromise(endpoint, { includeAllArgs: true, method: isNew ? 'POST' : 'PUT', data: rule, query: { duplicateRule: this.isDuplicateRule ? 'true' : 'false', wizardV3: 'true', }, }); // if we get a 202 back it means that we have an async task // running to lookup and verify the channel id for Slack. if (resp?.status === 202) { this.setState({detailedError: null, loading: true, uuid: data.uuid}); this.fetchStatus(); addLoadingMessage(t('Looking through all your channels...')); } else { this.handleRuleSuccess(isNew, data); } } catch (err) { this.setState({ detailedError: err.responseJSON || {__all__: 'Unknown error'}, loading: false, }); this.handleRuleSaveFailure(t('An error occurred')); } }; handleDeleteRule = async () => { const {project, rule} = this.state; const ruleId = isSavedAlertRule(rule) ? `${rule.id}/` : ''; const isNew = !ruleId; const {organization} = this.props; if (isNew) { return; } const endpoint = `/projects/${organization.slug}/${project.slug}/rules/${ruleId}`; addLoadingMessage(t('Deleting...')); try { await this.api.requestPromise(endpoint, { method: 'DELETE', }); addSuccessMessage(t('Deleted alert rule')); browserHistory.replace( recreateRoute('', { ...this.props, params: {...this.props.params, orgId: organization.slug}, stepBack: -2, }) ); } catch (err) { this.setState({ detailedError: err.responseJSON || {__all__: 'Unknown error'}, }); addErrorMessage(t('There was a problem deleting the alert')); } }; handleCancel = () => { const {organization, router} = this.props; router.push(normalizeUrl(`/organizations/${organization.slug}/alerts/rules/`)); }; hasError = (field: string) => { const {detailedError} = this.state; if (!detailedError) { return false; } return detailedError.hasOwnProperty(field); }; handleEnvironmentChange = (val: string) => { // If 'All Environments' is selected the value should be null if (val === ALL_ENVIRONMENTS_KEY) { this.handleChange('environment', null); } else { this.handleChange('environment', val); } }; handleChange = (prop: T, val: IssueAlertRule[T]) => { this.setState(prevState => { const clonedState = cloneDeep(prevState); set(clonedState, `rule[${prop}]`, val); return {...clonedState, detailedError: omit(prevState.detailedError, prop)}; }); }; handlePropertyChange = ( type: ConditionOrActionProperty, idx: number, prop: T, val: IssueAlertRuleAction[T] ) => { this.setState(prevState => { const clonedState = cloneDeep(prevState); set(clonedState, `rule[${type}][${idx}][${prop}]`, val); return clonedState; }); }; getInitialValue = (type: ConditionOrActionProperty, id: string) => { const configuration = this.state.configs?.[type]?.find(c => c.id === id); const hasChangeAlerts = configuration?.id && this.props.organization.features.includes('change-alerts') && CHANGE_ALERT_CONDITION_IDS.includes(configuration.id); return configuration?.formFields ? Object.fromEntries( Object.entries(configuration.formFields) // TODO(ts): Doesn't work if I cast formField as IssueAlertRuleFormField .map(([key, formField]: [string, any]) => [ key, hasChangeAlerts && key === 'interval' ? '1h' : formField?.initial ?? formField?.choices?.[0]?.[0], ]) .filter(([, initial]) => !!initial) ) : {}; }; handleResetRow = ( type: ConditionOrActionProperty, idx: number, prop: T, val: IssueAlertRuleAction[T] ) => { this.setState(prevState => { const clonedState = cloneDeep(prevState); // Set initial configuration, but also set const id = (clonedState.rule as IssueAlertRule)[type][idx].id; const newRule = { ...this.getInitialValue(type, id), id, [prop]: val, }; set(clonedState, `rule[${type}][${idx}]`, newRule); return clonedState; }); }; handleAddRow = ( type: ConditionOrActionProperty, item: IssueAlertRuleActionTemplate ) => { this.setState(prevState => { const clonedState = cloneDeep(prevState); // Set initial configuration const newRule = { ...this.getInitialValue(type, item.id), id: item.id, sentryAppInstallationUuid: item.sentryAppInstallationUuid, }; const newTypeList = prevState.rule ? prevState.rule[type] : []; set(clonedState, `rule[${type}]`, [...newTypeList, newRule]); return clonedState; }); const {organization} = this.props; const {project} = this.state; trackAnalytics('edit_alert_rule.add_row', { organization, project_id: project.id, type, name: item.id, }); }; handleDeleteRow = (type: ConditionOrActionProperty, idx: number) => { this.setState(prevState => { const clonedState = cloneDeep(prevState); const newTypeList = prevState.rule ? [...prevState.rule[type]] : []; newTypeList.splice(idx, 1); set(clonedState, `rule[${type}]`, newTypeList); return clonedState; }); }; handleAddCondition = (template: IssueAlertRuleActionTemplate) => this.handleAddRow('conditions', template); handleAddAction = (template: IssueAlertRuleActionTemplate) => this.handleAddRow('actions', template); handleAddFilter = (template: IssueAlertRuleActionTemplate) => this.handleAddRow('filters', template); handleDeleteCondition = (ruleIndex: number) => this.handleDeleteRow('conditions', ruleIndex); handleDeleteAction = (ruleIndex: number) => this.handleDeleteRow('actions', ruleIndex); handleDeleteFilter = (ruleIndex: number) => this.handleDeleteRow('filters', ruleIndex); handleChangeConditionProperty = (ruleIndex: number, prop: string, val: string) => this.handlePropertyChange('conditions', ruleIndex, prop, val); handleChangeActionProperty = (ruleIndex: number, prop: string, val: string) => this.handlePropertyChange('actions', ruleIndex, prop, val); handleChangeFilterProperty = (ruleIndex: number, prop: string, val: string) => this.handlePropertyChange('filters', ruleIndex, prop, val); handleResetCondition = (ruleIndex: number, prop: string, value: string) => this.handleResetRow('conditions', ruleIndex, prop, value); handleResetAction = (ruleIndex: number, prop: string, value: string) => this.handleResetRow('actions', ruleIndex, prop, value); handleResetFilter = (ruleIndex: number, prop: string, value: string) => this.handleResetRow('filters', ruleIndex, prop, value); handleValidateRuleName = () => { const isRuleNameEmpty = !this.state.rule?.name.trim(); if (!isRuleNameEmpty) { return; } this.setState(prevState => ({ detailedError: { ...prevState.detailedError, name: [t('Field Required')], }, })); }; getConditions() { const {organization} = this.props; if (!organization.features.includes('change-alerts')) { return this.state.configs?.conditions ?? null; } return ( this.state.configs?.conditions?.map(condition => CHANGE_ALERT_CONDITION_IDS.includes(condition.id) ? ({ ...condition, label: CHANGE_ALERT_PLACEHOLDERS_LABELS[condition.id], } as IssueAlertRuleConditionTemplate) : condition ) ?? null ); } getTeamId = () => { const {rule} = this.state; const owner = rule?.owner; // ownership follows the format team:, just grab the id return owner && owner.split(':')[1]; }; handleOwnerChange = ({value}: {value: string}) => { const ownerValue = value && `team:${value}`; this.handleChange('owner', ownerValue); }; renderLoading() { return this.renderBody(); } renderError() { return ( {t( 'Unable to access this alert rule -- check to make sure you have the correct permissions' )} ); } renderRuleName(disabled: boolean) { const {rule, detailedError} = this.state; const {name} = rule || {}; return ( ) => this.handleChange('name', event.target.value) } onBlur={this.handleValidateRuleName} disabled={disabled} /> ); } renderTeamSelect(disabled: boolean) { const {rule, project} = this.state; const ownerId = rule?.owner?.split(':')[1]; return ( team.isMember || team.id === ownerId || team.access.includes('team:admin') } useId includeUnassigned disabled={disabled} /> ); } renderIdBadge(project: Project) { return ( ); } renderEnvironmentSelect(disabled: boolean) { const {environments, rule} = this.state; const environmentOptions = [ { value: ALL_ENVIRONMENTS_KEY, label: t('All Environments'), }, ...(environments?.map(env => ({value: env.name, label: getDisplayName(env)})) ?? []), ]; const environment = !rule || !rule.environment ? ALL_ENVIRONMENTS_KEY : rule.environment; return ( {({onChange, onBlur}) => ( { this.handleEnvironmentChange(value); onChange(value, {}); onBlur(value, {}); }} /> )} ); } renderPreviewText() { const {issueCount, previewError} = this.state; if (previewError) { return t( "Select a condition above to see which issues would've triggered this alert" ); } return tct( "[issueCount] issues would have triggered this rule in the past 14 days [approximately:approximately]. If you're looking to reduce noise then make sure to [link:read the docs].", { issueCount, approximately: ( ), link: , } ); } renderPreviewTable() { const {members} = this.props; const { previewGroups, previewError, pageLinks, issueCount, previewPage, loadingPreview, } = this.state; return ( ); } renderProjectSelect(disabled: boolean) { const {project: _selectedProject, projects, organization} = this.props; const {rule} = this.state; const projectOptions = getProjectOptions({ organization, projects, isFormDisabled: disabled, }); return ( {({onChange, onBlur, model}) => { const selectedProject = projects.find(({id}) => id === model.getValue('projectId')) || _selectedProject; return ( ({ ...provided, marginBottom: `${space(1)}`, }), }} options={projectOptions} onChange={({value}: {value: Project['id']}) => { // if the current owner/team isn't part of project selected, update to the first available team const nextSelectedProject = projects.find(({id}) => id === value) ?? selectedProject; const ownerId: string | undefined = model .getValue('owner') ?.split(':')[1]; if ( ownerId && nextSelectedProject.teams.find(({id}) => id === ownerId) === undefined && nextSelectedProject.teams.length ) { this.handleOwnerChange({value: nextSelectedProject.teams[0].id}); } this.setState({project: nextSelectedProject}); onChange(value, {}); onBlur(value, {}); }} components={{ SingleValue: containerProps => ( ), }} /> ); }} ); } renderActionInterval(disabled: boolean) { const {rule} = this.state; const {frequency} = rule || {}; return ( {({onChange, onBlur}) => ( { this.handleChange('frequency', value); onChange(value, {}); onBlur(value, {}); }} /> )} ); } renderBody() { const {organization} = this.props; const { project, rule, detailedError, loading, ownership, sendingNotification, incompatibleConditions, incompatibleFilters, } = this.state; const {actions, filters, conditions, frequency} = rule || {}; const environment = !rule || !rule.environment ? ALL_ENVIRONMENTS_KEY : rule.environment; const canCreateAlert = hasEveryAccess(['alerts:write'], {organization, project}); const disabled = loading || !(canCreateAlert || isActiveSuperuser()); // Note `key` on `
` below is so that on initial load, we show // the form with a loading mask on top of it, but force a re-render by using // a different key when we have fetched the rule so that form inputs are filled in return (
) : null } > {loading && } {t('Select an environment and project')} {this.renderEnvironmentSelect(disabled)} {this.renderProjectSelect(disabled)} {t('Set conditions')} {tct( '[when:When] an event is captured by Sentry and [selector] of the following happens', { when: , selector: ( ({ ...provided, minHeight: '21px', height: '21px', }), }} inline={false} isSearchable={false} isClearable={false} name="actionMatch" required flexibleControlStateSize options={ACTION_MATCH_OPTIONS_MIGRATED} onChange={val => this.handleChange('actionMatch', val) } size="xs" disabled={disabled} /> ), } )} {detailedError?.conditions[0]} {(detailedError?.conditions[0] || '').startsWith( 'You may not exceed' ) && ( {' '} {t('View Docs')} )} ) } incompatibleRules={incompatibleConditions} incompatibleBanner={ incompatibleFilters === null && incompatibleConditions !== null ? incompatibleConditions.at(-1) : null } /> {tct('[if:If][selector] of these filters match', { if: , selector: ( ({ ...provided, minHeight: '21px', height: '21px', }), }} inline={false} isSearchable={false} isClearable={false} name="filterMatch" required flexibleControlStateSize options={ACTION_MATCH_OPTIONS} onChange={val => this.handleChange('filterMatch', val)} size="xs" disabled={disabled} /> ), })} {detailedError?.filters[0]} ) } incompatibleRules={incompatibleFilters} incompatibleBanner={ incompatibleFilters ? incompatibleFilters.at(-1) : null } /> {tct('[then:Then] perform these actions', { then: , })} {detailedError?.actions[0]} ) } /> {t('Set action interval')} {t('Perform the actions above once this often for an issue')} {this.renderActionInterval(disabled)}
{t('Preview')} {this.renderPreviewText()}
{this.renderPreviewTable()} {t('Add a name and owner')} {t( 'This name will show up in notifications and the owner will give permissions to your whole team to edit and view this alert.' )} {this.renderRuleName(disabled)} {this.renderTeamSelect(disabled)}
); } } export default withOrganization(withProjects(IssueRuleEditor)); export const findIncompatibleRules = ( rule: IssueAlertRule | UnsavedIssueAlertRule | null | undefined ): IncompatibleRule => { if (!rule) { return {conditionIndices: null, filterIndices: null}; } const {conditions, filters} = rule; // Check for more than one 'issue state change' condition // or 'FirstSeenEventCondition' + 'EventFrequencyCondition' if (rule.actionMatch === 'all') { let firstSeen = -1; let regression = -1; let reappeared = -1; let eventFrequency = -1; let userFrequency = -1; for (let i = 0; i < conditions.length; i++) { const id = conditions[i].id; if (id.endsWith('FirstSeenEventCondition')) { firstSeen = i; } else if (id.endsWith('RegressionEventCondition')) { regression = i; } else if (id.endsWith('ReappearedEventCondition')) { reappeared = i; } else if ( id.endsWith('EventFrequencyCondition') && (conditions[i].value as number) >= 1 ) { eventFrequency = i; } else if ( id.endsWith('EventUniqueUserFrequencyCondition') && (conditions[i].value as number) >= 1 ) { userFrequency = i; } // FirstSeenEventCondition is incompatible with all the following types const firstSeenError = firstSeen !== -1 && [regression, reappeared, eventFrequency, userFrequency].some(idx => idx !== -1); const regressionReappearedError = regression !== -1 && reappeared !== -1; if (firstSeenError || regressionReappearedError) { const indices = [firstSeen, regression, reappeared, eventFrequency, userFrequency] .filter(idx => idx !== -1) .sort((a, b) => a - b); return {conditionIndices: indices, filterIndices: null}; } } } // Check for 'FirstSeenEventCondition' and ('IssueOccurrencesFilter' or 'AgeComparisonFilter') // Considers the case where filterMatch is 'any' and all filters are incompatible const firstSeen = conditions.findIndex(condition => condition.id.endsWith('FirstSeenEventCondition') ); if (firstSeen !== -1 && (rule.actionMatch === 'all' || conditions.length === 1)) { let incompatibleFilters = 0; for (let i = 0; i < filters.length; i++) { const filter = filters[i]; const id = filter.id; if (id.endsWith('IssueOccurrencesFilter') && filter) { if ( (rule.filterMatch === 'all' && (filter.value as number) > 1) || (rule.filterMatch === 'none' && (filter.value as number) <= 1) ) { return {conditionIndices: [firstSeen], filterIndices: [i]}; } if (rule.filterMatch === 'any' && (filter.value as number) > 1) { incompatibleFilters += 1; } } else if (id.endsWith('AgeComparisonFilter')) { if (rule.filterMatch !== 'none') { if (filter.comparison_type === 'older') { if (rule.filterMatch === 'all') { return {conditionIndices: [firstSeen], filterIndices: [i]}; } incompatibleFilters += 1; } } else if (filter.comparison_type === 'newer' && (filter.value as number) > 0) { return {conditionIndices: [firstSeen], filterIndices: [i]}; } } } if (incompatibleFilters === filters.length && incompatibleFilters > 0) { return { conditionIndices: [firstSeen], filterIndices: [...Array(filters.length).keys()], }; } } return {conditionIndices: null, filterIndices: null}; }; // TODO(ts): Understand why styled is not correctly inheriting props here const StyledForm = styled(Form)` position: relative; `; const ConditionsPanel = styled(Panel)` padding-top: ${space(0.5)}; padding-bottom: ${space(2)}; `; const StyledAlert = styled(Alert)` margin-bottom: 0; `; const StyledListItem = styled(ListItem)` margin: ${space(2)} 0 ${space(1)} 0; font-size: ${p => p.theme.fontSizeExtraLarge}; `; const StyledListItemSpaced = styled('div')` display: flex; justify-content: space-between; `; const StyledFieldHelp = styled(FieldHelp)` margin-top: 0; @media (max-width: ${p => p.theme.breakpoints.small}) { margin-left: -${space(4)}; } `; const SetConditionsListItem = styled(StyledListItem)` display: flex; justify-content: space-between; `; const Step = styled('div')` position: relative; display: flex; align-items: flex-start; margin: ${space(4)} ${space(4)} ${space(3)} ${space(1)}; `; const StepHeader = styled('h5')` margin-bottom: ${space(1)}; `; const StepContainer = styled('div')` position: relative; display: flex; align-items: flex-start; flex-grow: 1; `; const StepContent = styled('div')` flex-grow: 1; `; const StepConnector = styled('div')` position: absolute; height: 100%; top: 28px; left: 19px; border-right: 1px ${p => p.theme.gray200} dashed; `; const StepLead = styled('div')` margin-bottom: ${space(0.5)}; display: flex; align-items: center; gap: ${space(0.5)}; `; const TestButtonWrapper = styled('div')` margin-top: ${space(1.5)}; `; const ChevronContainer = styled('div')` display: flex; align-items: center; padding: ${space(0.5)} ${space(1.5)}; `; const Badge = styled('span')` min-width: 56px; background-color: ${p => p.theme.purple300}; padding: 0 ${space(0.75)}; border-radius: ${p => p.theme.borderRadius}; color: ${p => p.theme.white}; text-transform: uppercase; text-align: center; font-size: ${p => p.theme.fontSizeMedium}; font-weight: 600; line-height: 1.5; `; const EmbeddedWrapper = styled('div')` width: 80px; `; const EmbeddedSelectField = styled(SelectField)` padding: 0; font-weight: normal; text-transform: none; `; const SemiTransparentLoadingMask = styled(LoadingMask)` opacity: 0.6; z-index: 1; /* Needed so that it sits above form elements */ `; const SettingsContainer = styled('div')` display: grid; grid-template-columns: 1fr 1fr; gap: ${space(1)}; `; const StyledField = styled(FieldGroup)` border-bottom: none; padding: 0; & > div { padding: 0; width: 100%; } margin-bottom: ${space(1)}; `; const StyledFieldWrapper = styled('div')` @media (min-width: ${p => p.theme.breakpoints.small}) { display: grid; grid-template-columns: 2fr 1fr; gap: ${space(1)}; } margin-bottom: 60px; `; const ContentIndent = styled('div')` @media (min-width: ${p => p.theme.breakpoints.small}) { margin-left: ${space(4)}; } `; const Main = styled(Layout.Main)` padding: ${space(2)} ${space(4)}; `;