import {Fragment, useRef} from 'react'; import {browserHistory} from 'react-router'; import {css} from '@emotion/react'; import styled from '@emotion/styled'; import {Observer} from 'mobx-react'; import NumberField from 'sentry/components/forms/fields/numberField'; import SelectField from 'sentry/components/forms/fields/selectField'; import SentryProjectSelectorField from 'sentry/components/forms/fields/sentryProjectSelectorField'; import TextField from 'sentry/components/forms/fields/textField'; import Form from 'sentry/components/forms/form'; import FormModel from 'sentry/components/forms/model'; import Panel from 'sentry/components/panels/panel'; import PanelBody from 'sentry/components/panels/panelBody'; import {timezoneOptions} from 'sentry/data/timezones'; import {t} from 'sentry/locale'; import HookStore from 'sentry/stores/hookStore'; import {space} from 'sentry/styles/space'; import {isActiveSuperuser} from 'sentry/utils/isActiveSuperuser'; import commonTheme from 'sentry/utils/theme'; import useOrganization from 'sentry/utils/useOrganization'; import usePageFilters from 'sentry/utils/usePageFilters'; import useProjects from 'sentry/utils/useProjects'; import {normalizeUrl} from 'sentry/utils/withDomainRequired'; import {MockTimelineVisualization} from 'sentry/views/monitors/components/mockTimelineVisualization'; import { DEFAULT_CRONTAB, DEFAULT_MONITOR_TYPE, mapMonitorFormErrors, transformMonitorFormData, } from 'sentry/views/monitors/components/monitorForm'; import type {Monitor} from 'sentry/views/monitors/types'; import {ScheduleType} from 'sentry/views/monitors/types'; import {getScheduleIntervals} from 'sentry/views/monitors/utils'; import {crontabAsText} from 'sentry/views/monitors/utils/crontabAsText'; const DEFAULT_SCHEDULE_CONFIG = { scheduleType: 'crontab', cronSchedule: DEFAULT_CRONTAB, intervalFrequency: '1', intervalUnit: 'day', }; export default function MonitorCreateForm() { const organization = useOrganization(); const {projects} = useProjects(); const {selection} = usePageFilters(); const monitorCreationCallbacks = HookStore.get('callback:on-monitor-created'); const form = useRef( new FormModel({ transformData: transformMonitorFormData, mapFormErrors: mapMonitorFormErrors, }) ); const selectedProjectId = selection.projects[0]; const selectedProject = selectedProjectId ? projects.find(p => p.id === selectedProjectId + '') : null; const isSuperuser = isActiveSuperuser(); const filteredProjects = projects.filter(project => isSuperuser || project.isMember); function onCreateMonitor(data: Monitor) { const endpointOptions = { query: { project: selection.projects, environment: selection.environments, }, }; browserHistory.push( normalizeUrl({ pathname: `/organizations/${organization.slug}/crons/${data.project.slug}/${data.slug}/`, query: endpointOptions.query, }) ); monitorCreationCallbacks.map(cb => cb(organization)); } function changeScheduleType(type: ScheduleType) { form.current.setValue('config.schedule_type', type); } return (
{t('SCHEDULE')} {() => { const currScheduleType = form.current.getValue('config.schedule_type'); const selectedCrontab = currScheduleType === ScheduleType.CRONTAB; const parsedSchedule = form.current.getError('config.schedule') ? '' : crontabAsText( form.current.getValue('config.schedule')?.toString() ?? '' ); return ( changeScheduleType(ScheduleType.CRONTAB)} > {t('Crontab Schedule')} {parsedSchedule} changeScheduleType(ScheduleType.INTERVAL)} > {t('Interval Schedule')} ); }} {() => { const scheduleType = form.current.getValue('config.schedule_type'); const cronSchedule = form.current.getValue('config.schedule'); const intervalFrequency = form.current.getValue('config.schedule.frequency'); const intervalUnit = form.current.getValue('config.schedule.interval'); const schedule = { scheduleType, cronSchedule, intervalFrequency, intervalUnit, }; return ; }}
); } const FieldContainer = styled('div')` width: 800px; `; const SchedulePanel = styled(Panel)<{highlighted: boolean}>` border-radius: 0 ${space(0.75)} ${space(0.75)} 0; ${p => p.highlighted ? css` border: 2px solid ${p.theme.purple300}; ` : css` padding: 1px; `}; &:first-child { border-radius: ${space(0.75)} 0 0 ${space(0.75)}; } `; const ScheduleLabel = styled('div')` font-weight: bold; margin-bottom: ${space(2)}; `; const Label = styled('div')` font-weight: bold; color: ${p => p.theme.subText}; `; const LabelText = styled(Label)` margin-top: ${space(2)}; margin-bottom: ${space(1)}; `; const ScheduleOptions = styled('div')` display: grid; grid-template-columns: 1fr 1fr; `; const MultiColumnInput = styled('div')<{columns?: string}>` display: grid; align-items: center; gap: ${space(1)}; grid-template-columns: ${p => p.columns}; `; const CronstrueText = styled(LabelText)` font-weight: normal; font-size: ${p => p.theme.fontSizeExtraSmall}; font-family: ${p => p.theme.text.familyMono}; grid-column: auto / span 2; `; const StyledNumberField = styled(NumberField)` padding: 0; `; const StyledSelectField = styled(SelectField)` padding: 0; `; const StyledTextField = styled(TextField)` padding: 0; `; const StyledSentryProjectSelectorField = styled(SentryProjectSelectorField)` padding: 0; `;