import {Component, Fragment} from 'react'; import {RouteComponentProps} from 'react-router'; import styled from '@emotion/styled'; import * as Layout from 'sentry/components/layouts/thirds'; import LoadingIndicator from 'sentry/components/loadingIndicator'; import SentryDocumentTitle from 'sentry/components/sentryDocumentTitle'; import {t} from 'sentry/locale'; import {Member, Organization, Project} from 'sentry/types'; import EventView from 'sentry/utils/discover/eventView'; import {uniqueId} from 'sentry/utils/guid'; import withRouteAnalytics, { WithRouteAnalyticsProps, } from 'sentry/utils/routeAnalytics/withRouteAnalytics'; import Teams from 'sentry/utils/teams'; import {normalizeUrl} from 'sentry/utils/withDomainRequired'; import BuilderBreadCrumbs from 'sentry/views/alerts/builder/builderBreadCrumbs'; import IssueRuleEditor from 'sentry/views/alerts/rules/issue'; import MetricRulesCreate from 'sentry/views/alerts/rules/metric/create'; import MetricRulesDuplicate from 'sentry/views/alerts/rules/metric/duplicate'; import {AlertRuleType} from 'sentry/views/alerts/types'; import { AlertType as WizardAlertType, AlertWizardAlertNames, DEFAULT_WIZARD_TEMPLATE, WizardRuleTemplate, } from 'sentry/views/alerts/wizard/options'; import {getAlertTypeFromAggregateDataset} from 'sentry/views/alerts/wizard/utils'; type RouteParams = { alertType?: AlertRuleType; projectId?: string; }; type Props = RouteComponentProps & WithRouteAnalyticsProps & { hasMetricAlerts: boolean; members: Member[] | undefined; organization: Organization; project: Project; }; type State = { alertType: AlertRuleType; }; class Create extends Component { state = this.getInitialState(); getInitialState(): State { const {organization, location, project, params, router} = this.props; const {aggregate, dataset, eventTypes, createFromDuplicate} = location?.query ?? {}; const alertType = params.alertType || AlertRuleType.METRIC; // TODO(taylangocmen): Remove redirect with aggregate && dataset && eventTypes, init from template if ( alertType === AlertRuleType.METRIC && !(aggregate && dataset && eventTypes) && !createFromDuplicate ) { router.replace( normalizeUrl({ ...location, pathname: `/organizations/${organization.slug}/alerts/new/${alertType}`, query: { ...location.query, ...DEFAULT_WIZARD_TEMPLATE, project: project.slug, }, }) ); } return {alertType}; } componentDidMount() { const {project} = this.props; this.props.setRouteAnalyticsParams({ project_id: project.id, session_id: this.sessionId, alert_type: this.state.alertType, duplicate_rule: this.isDuplicateRule ? 'true' : 'false', wizard_v3: 'true', }); this.props.setEventNames('new_alert_rule.viewed', 'New Alert Rule: Viewed'); } /** Used to track analytics within one visit to the creation page */ sessionId = uniqueId(); get isDuplicateRule(): boolean { const {location} = this.props; const createFromDuplicate = location?.query.createFromDuplicate === 'true'; return createFromDuplicate && location?.query.duplicateRuleId; } render() { const {hasMetricAlerts, organization, project, location, routes, members} = this.props; const {alertType} = this.state; const {aggregate, dataset, eventTypes, createFromWizard, createFromDiscover} = location?.query ?? {}; const wizardTemplate: WizardRuleTemplate = { aggregate: aggregate ?? DEFAULT_WIZARD_TEMPLATE.aggregate, dataset: dataset ?? DEFAULT_WIZARD_TEMPLATE.dataset, eventTypes: eventTypes ?? DEFAULT_WIZARD_TEMPLATE.eventTypes, }; const eventView = createFromDiscover ? EventView.fromLocation(location) : undefined; let wizardAlertType: undefined | WizardAlertType; if (createFromWizard && alertType === AlertRuleType.METRIC) { wizardAlertType = wizardTemplate ? getAlertTypeFromAggregateDataset(wizardTemplate) : 'issues'; } const title = t('New Alert Rule'); return ( {wizardAlertType ? `${t('Set Conditions for')} ${AlertWizardAlertNames[wizardAlertType]}` : title} {({teams, initiallyLoaded}) => initiallyLoaded ? ( {(!hasMetricAlerts || alertType === AlertRuleType.ISSUE) && ( id)} members={members} /> )} {hasMetricAlerts && alertType === AlertRuleType.METRIC && (this.isDuplicateRule ? ( id)} /> ) : ( id)} /> ))} ) : ( ) } ); } } const Body = styled(Layout.Body)` && { padding: 0; gap: 0; } grid-template-rows: 1fr; @media (min-width: ${p => p.theme.breakpoints.large}) { grid-template-columns: minmax(100px, auto) 400px; } `; export default withRouteAnalytics(Create);