From 8ff21963d34dbdfcb7fc44cf25b1ac2a68881832 Mon Sep 17 00:00:00 2001 From: Colleen O'Rourke Date: Mon, 6 Jan 2025 11:55:21 -0800 Subject: [PATCH] chore(alerts): Remove fe code for activated alerts (#81218) MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit Remove the front end for activated alerts as it's being deprecated. I can't merge this until we clean up a couple live alerts but I wanted to get the PRs ready. The gigantic backend counterpart is here https://github.com/getsentry/sentry/pull/81095 **Before** Screenshot 2024-11-25 at 1 25 54 PM **After** Screenshot 2024-11-25 at 1 27 26 PM --- .../rules/metric/ruleConditionsForm.tsx | 189 +----------------- .../alerts/rules/metric/ruleForm.spec.tsx | 96 --------- .../views/alerts/rules/metric/ruleForm.tsx | 66 +----- 3 files changed, 2 insertions(+), 349 deletions(-) diff --git a/static/app/views/alerts/rules/metric/ruleConditionsForm.tsx b/static/app/views/alerts/rules/metric/ruleConditionsForm.tsx index 6c0e69418a0733..08455e1fdf2d20 100644 --- a/static/app/views/alerts/rules/metric/ruleConditionsForm.tsx +++ b/static/app/views/alerts/rules/metric/ruleConditionsForm.tsx @@ -36,7 +36,6 @@ import {SearchQueryBuilder} from 'sentry/components/searchQueryBuilder'; import {InvalidReason} from 'sentry/components/searchSyntax/parser'; import {t, tct} from 'sentry/locale'; import {space} from 'sentry/styles/space'; -import {ActivationConditionType, MonitorType} from 'sentry/types/alerts'; import type {SelectValue} from 'sentry/types/core'; import type {Tag, TagCollection} from 'sentry/types/group'; import type {InjectedRouter} from 'sentry/types/legacyReactRouter'; @@ -105,12 +104,6 @@ type Props = { isEditing: boolean; onComparisonDeltaChange: (value: number) => void; onFilterSearch: (query: string, isQueryValid) => void; - onMonitorTypeSelect: (activatedAlertFields: { - activationCondition?: ActivationConditionType | undefined; - monitorType?: MonitorType; - monitorWindowSuffix?: string | undefined; - monitorWindowValue?: number | undefined; - }) => void; onTimeWindowChange: (value: number) => void; organization: Organization; project: Project; @@ -120,7 +113,6 @@ type Props = { thresholdChart: React.ReactNode; timeWindow: number; // optional props - activationCondition?: ActivationConditionType; allowChangeEventTypes?: boolean; comparisonDelta?: number; disableProjectSelector?: boolean; @@ -130,7 +122,6 @@ type Props = { isLowConfidenceChartData?: boolean; isTransactionMigration?: boolean; loadingProjects?: boolean; - monitorType?: number; }; type State = { @@ -499,16 +490,7 @@ class RuleConditionsForm extends PureComponent { } renderInterval() { - const { - organization, - disabled, - alertType, - timeWindow, - onTimeWindowChange, - project, - monitorType, - isForLlmMetric, - } = this.props; + const {organization, disabled, alertType, project, isForLlmMetric} = this.props; return ( @@ -536,118 +518,6 @@ class RuleConditionsForm extends PureComponent { required /> )} - - {monitorType !== MonitorType.ACTIVATED && ( - onTimeWindowChange(value)} - inline={false} - flexibleControlStateSize - /> - )} - - - ); - } - - renderMonitorTypeSelect() { - // TODO: disable select on edit - const { - activationCondition, - isEditing, - monitorType, - onMonitorTypeSelect, - onTimeWindowChange, - timeWindow, - } = this.props; - - return ( - - - -
{t('Select Monitor Type')}
-
-
- - - - isEditing - ? null - : onMonitorTypeSelect({ - monitorType: MonitorType.CONTINUOUS, - }) - } - > - {t('Continuous')} -
{t('Continuously monitor trends for the metrics outlined below')}
-
- - isEditing - ? null - : onMonitorTypeSelect({ - monitorType: MonitorType.ACTIVATED, - }) - } - > - Conditional - {monitorType === MonitorType.ACTIVATED ? ( - - {`${t('Monitor')} `} - - onMonitorTypeSelect({activationCondition: value}) - } - inline={false} - flexibleControlStateSize - size="xs" - /> - {` ${t('for')} `} - onTimeWindowChange(value)} - inline={false} - flexibleControlStateSize - size="xs" - /> - - ) : ( -
- {t('Temporarily monitor specified query given activation condition')} -
- )} -
-
); @@ -671,8 +541,6 @@ class RuleConditionsForm extends PureComponent { } = this.props; const {environments, filterKeys} = this.state; - const hasActivatedAlerts = organization.features.includes('activated-alert-rules'); - const environmentOptions: SelectValue[] = [ { value: null, @@ -718,7 +586,6 @@ class RuleConditionsForm extends PureComponent { )} )} - {hasActivatedAlerts && this.renderMonitorTypeSelect()} {!isErrorMigration && this.renderInterval()} {t('Filter events')} @@ -943,58 +810,4 @@ const FormRow = styled('div')<{columns?: number; noMargin?: boolean}>` `} `; -const MonitorSelect = styled('div')` - border-radius: ${p => p.theme.borderRadius}; - border: 1px solid ${p => p.theme.border}; - width: 100%; - display: grid; - grid-template-columns: 1fr 1fr; - height: 5rem; -`; - -type MonitorCardProps = { - isSelected: boolean; - /** - * Adds hover and focus states to the card - */ - position: 'left' | 'right'; - disabled?: boolean; -}; - -const MonitorCard = styled('div')` - padding: ${space(1)} ${space(2)}; - display: flex; - flex-grow: 1; - flex-direction: column; - cursor: ${p => (p.disabled || p.isSelected ? 'default' : 'pointer')}; - justify-content: center; - background-color: ${p => - p.disabled && !p.isSelected ? p.theme.backgroundSecondary : p.theme.background}; - - &:focus, - &:hover { - ${p => - p.disabled || p.isSelected - ? '' - : ` - outline: 1px solid ${p.theme.purple200}; - background-color: ${p.theme.backgroundSecondary}; - `} - } - - border-top-left-radius: ${p => (p.position === 'left' ? p.theme.borderRadius : 0)}; - border-bottom-left-radius: ${p => (p.position === 'left' ? p.theme.borderRadius : 0)}; - border-top-right-radius: ${p => (p.position !== 'left' ? p.theme.borderRadius : 0)}; - border-bottom-right-radius: ${p => (p.position !== 'left' ? p.theme.borderRadius : 0)}; - margin: ${p => - p.isSelected ? (p.position === 'left' ? '1px 2px 1px 0' : '1px 0 1px 2px') : 0}; - outline: ${p => (p.isSelected ? `2px solid ${p.theme.purple400}` : 'none')}; -`; - -const ActivatedAlertFields = styled('div')` - display: flex; - align-items: center; - justify-content: space-between; -`; - export default withApi(withProjects(withTags(RuleConditionsForm))); diff --git a/static/app/views/alerts/rules/metric/ruleForm.spec.tsx b/static/app/views/alerts/rules/metric/ruleForm.spec.tsx index e53a2609b1c857..b1a50d37daefc9 100644 --- a/static/app/views/alerts/rules/metric/ruleForm.spec.tsx +++ b/static/app/views/alerts/rules/metric/ruleForm.spec.tsx @@ -9,7 +9,6 @@ import selectEvent from 'sentry-test/selectEvent'; import {addErrorMessage} from 'sentry/actionCreators/indicator'; import type FormModel from 'sentry/components/forms/model'; import ProjectsStore from 'sentry/stores/projectsStore'; -import {ActivationConditionType, MonitorType} from 'sentry/types/alerts'; import {metric} from 'sentry/utils/analytics'; import RuleFormContainer from 'sentry/views/alerts/rules/metric/ruleForm'; import { @@ -151,23 +150,6 @@ describe('Incident Rules Form', () => { expect(await screen.findByLabelText('Save Rule')).toBeEnabled(); expect(screen.queryByText(permissionAlertText)).not.toBeInTheDocument(); }); - - it('renders time window', async () => { - createWrapper({rule}); - - expect(await screen.findByText('1 hour interval')).toBeInTheDocument(); - }); - - it('renders time window for activated alerts', async () => { - createWrapper({ - rule: { - ...rule, - monitorType: MonitorType.CONTINUOUS, - }, - }); - - expect(await screen.findByText('1 hour interval')).toBeInTheDocument(); - }); }); describe('Creating a new rule', () => { @@ -301,84 +283,6 @@ describe('Incident Rules Form', () => { ); }); - // Activation condition - it('creates a rule with an activation condition', async () => { - organization.features = [ - ...organization.features, - 'mep-rollout-flag', - 'activated-alert-rules', - ]; - const rule = MetricRuleFixture({ - monitorType: MonitorType.ACTIVATED, - activationCondition: ActivationConditionType.RELEASE_CREATION, - }); - createWrapper({ - rule: { - ...rule, - id: undefined, - aggregate: 'count()', - eventTypes: ['transaction'], - dataset: 'transactions', - }, - }); - - expect(await screen.findByTestId('alert-total-events')).toHaveTextContent('Total5'); - - await userEvent.click(screen.getByLabelText('Save Rule')); - - expect(createRule).toHaveBeenCalledWith( - expect.anything(), - expect.objectContaining({ - data: expect.objectContaining({ - name: 'My Incident Rule', - projects: ['project-slug'], - aggregate: 'count()', - eventTypes: ['transaction'], - dataset: 'generic_metrics', - thresholdPeriod: 1, - }), - }) - ); - }); - - it('creates a continuous rule with activated rules enabled', async () => { - organization.features = [ - ...organization.features, - 'mep-rollout-flag', - 'activated-alert-rules', - ]; - const rule = MetricRuleFixture({ - monitorType: MonitorType.CONTINUOUS, - }); - createWrapper({ - rule: { - ...rule, - id: undefined, - aggregate: 'count()', - eventTypes: ['transaction'], - dataset: 'transactions', - }, - }); - - expect(await screen.findByTestId('alert-total-events')).toHaveTextContent('Total5'); - - await userEvent.click(screen.getByLabelText('Save Rule')); - - expect(createRule).toHaveBeenCalledWith( - expect.anything(), - expect.objectContaining({ - data: expect.objectContaining({ - name: 'My Incident Rule', - projects: ['project-slug'], - aggregate: 'count()', - eventTypes: ['transaction'], - dataset: 'generic_metrics', - thresholdPeriod: 1, - }), - }) - ); - }); - it('creates an anomaly detection rule', async () => { organization.features = [ ...organization.features, diff --git a/static/app/views/alerts/rules/metric/ruleForm.tsx b/static/app/views/alerts/rules/metric/ruleForm.tsx index d6d97f11e40ebb..8192c7e6bec158 100644 --- a/static/app/views/alerts/rules/metric/ruleForm.tsx +++ b/static/app/views/alerts/rules/metric/ruleForm.tsx @@ -25,7 +25,6 @@ import ListItem from 'sentry/components/list/listItem'; import {t, tct} from 'sentry/locale'; import IndicatorStore from 'sentry/stores/indicatorStore'; import {space} from 'sentry/styles/space'; -import {ActivationConditionType, MonitorType} from 'sentry/types/alerts'; import type {PlainRoute, RouteComponentProps} from 'sentry/types/legacyReactRouter'; import type { EventsStats, @@ -153,13 +152,11 @@ type State = { timeWindow: number; triggerErrors: Map; triggers: Trigger[]; - activationCondition?: ActivationConditionType; chartError?: boolean; chartErrorMessage?: string; comparisonDelta?: number; isExtrapolatedChartData?: boolean; isLowConfidenceChartData?: boolean; - monitorType?: MonitorType; seasonality?: AlertRuleSeasonality; } & DeprecatedAsyncComponent['state']; @@ -208,7 +205,7 @@ class RuleFormContainer extends DeprecatedAsyncComponent { } getDefaultState(): State { - const {rule, location, organization} = this.props; + const {rule, location} = this.props; const triggersClone = [...rule.triggers]; const { aggregate: _aggregate, @@ -233,8 +230,6 @@ class RuleFormContainer extends DeprecatedAsyncComponent { ? `is:unresolved ${rule.query ?? ''}` : rule.query ?? ''; - const hasActivatedAlerts = organization.features.includes('activated-alert-rules'); - return { ...super.getDefaultState(), currentData: [], @@ -266,11 +261,6 @@ class RuleFormContainer extends DeprecatedAsyncComponent { project: this.props.project, owner: rule.owner, alertType: getAlertTypeFromAggregateDataset({aggregate, dataset}), - monitorType: hasActivatedAlerts - ? rule.monitorType || MonitorType.CONTINUOUS - : undefined, - activationCondition: - rule.activationCondition || ActivationConditionType.RELEASE_CREATION, }; } @@ -643,24 +633,6 @@ class RuleFormContainer extends DeprecatedAsyncComponent { this.setState({query, dataset, isQueryValid}); }; - handleMonitorTypeSelect = (activatedAlertFields: { - activationCondition?: ActivationConditionType | undefined; - monitorType?: MonitorType; - monitorWindowSuffix?: string | undefined; - monitorWindowValue?: number | undefined; - }) => { - const {monitorType} = activatedAlertFields; - let updatedFields = activatedAlertFields; - if (monitorType === MonitorType.CONTINUOUS) { - updatedFields = { - ...updatedFields, - activationCondition: undefined, - monitorWindowValue: undefined, - }; - } - this.setState(updatedFields as State); - }; - validateOnDemandMetricAlert() { if ( !isOnDemandMetricAlert(this.state.dataset, this.state.aggregate, this.state.query) @@ -671,18 +643,6 @@ class RuleFormContainer extends DeprecatedAsyncComponent { return !this.state.aggregate.includes(AggregationKey.PERCENTILE); } - validateActivatedAlerts() { - const {organization} = this.props; - const {monitorType, activationCondition, timeWindow} = this.state; - - const hasActivatedAlerts = organization.features.includes('activated-alert-rules'); - return ( - !hasActivatedAlerts || - monitorType !== MonitorType.ACTIVATED || - (activationCondition !== undefined && timeWindow) - ); - } - validateSubmit = model => { if (!this.validateMri()) { addErrorMessage(t('You need to select a metric before you can save the alert')); @@ -695,7 +655,6 @@ class RuleFormContainer extends DeprecatedAsyncComponent { const triggerErrors = this.validateTriggers(); const validTriggers = Array.from(triggerErrors).length === 0; const validOnDemandAlert = this.validateOnDemandMetricAlert(); - const validActivatedAlerts = this.validateActivatedAlerts(); if (!validTriggers) { this.setState(state => ({ @@ -721,13 +680,6 @@ class RuleFormContainer extends DeprecatedAsyncComponent { return false; } - if (!validActivatedAlerts) { - addErrorMessage( - t('Activation condition and monitor window must be set for activated alerts') - ); - return false; - } - return true; }; @@ -760,8 +712,6 @@ class RuleFormContainer extends DeprecatedAsyncComponent { comparisonDelta, timeWindow, eventTypes, - monitorType, - activationCondition, sensitivity, seasonality, comparisonType, @@ -772,7 +722,6 @@ class RuleFormContainer extends DeprecatedAsyncComponent { trigger.label !== AlertRuleTriggerType.WARNING || !isEmpty(trigger.alertThreshold) ); - const hasActivatedAlerts = organization.features.includes('activated-alert-rules'); // form model has all form state data, however we use local state to keep // track of the list of triggers (and actions within triggers) const loadingIndicator = IndicatorStore.addMessage( @@ -794,13 +743,6 @@ class RuleFormContainer extends DeprecatedAsyncComponent { metric.startSpan({name: 'saveAlertRule'}); - let activatedAlertFields = {}; - if (hasActivatedAlerts) { - activatedAlertFields = { - monitorType, - activationCondition, - }; - } const detectionTypes = new Map([ [AlertRuleComparisonType.COUNT, 'static'], [AlertRuleComparisonType.CHANGE, 'percent'], @@ -817,7 +759,6 @@ class RuleFormContainer extends DeprecatedAsyncComponent { { ...rule, // existing rule ...model.getTransformedData(), // form data - ...activatedAlertFields, projects: [project.slug], triggers: sanitizedTriggers, resolveThreshold: @@ -1305,8 +1246,6 @@ class RuleFormContainer extends DeprecatedAsyncComponent { isExtrapolatedChartData, isLowConfidenceChartData, triggersHaveChanged, - activationCondition, - monitorType, } = this.state; const wizardBuilderChart = this.renderTriggerChart(); @@ -1420,7 +1359,6 @@ class RuleFormContainer extends DeprecatedAsyncComponent { > { ].includes(aggregate)} isTransactionMigration={isMigration && !showErrorMigrationWarning} isLowConfidenceChartData={isLowConfidenceChartData} - monitorType={monitorType} onComparisonDeltaChange={value => this.handleFieldChange('comparisonDelta', value) } onFilterSearch={this.handleFilterUpdate} - onMonitorTypeSelect={this.handleMonitorTypeSelect} onTimeWindowChange={value => this.handleFieldChange('timeWindow', value)} organization={organization} project={project}