123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994 |
- import {Fragment, PureComponent} from 'react';
- import type {InjectedRouter} from 'react-router';
- import {components} from 'react-select';
- import {css} from '@emotion/react';
- import styled from '@emotion/styled';
- import omit from 'lodash/omit';
- import pick from 'lodash/pick';
- import {addErrorMessage} from 'sentry/actionCreators/indicator';
- import {fetchTagValues} from 'sentry/actionCreators/tags';
- import type {Client} from 'sentry/api';
- import {
- OnDemandMetricAlert,
- OnDemandWarningIcon,
- } from 'sentry/components/alerts/onDemandMetricAlert';
- import SearchBar, {getHasTag} from 'sentry/components/events/searchBar';
- import {
- STATIC_FIELD_TAGS,
- STATIC_FIELD_TAGS_WITHOUT_ERROR_FIELDS,
- STATIC_FIELD_TAGS_WITHOUT_TRACING,
- STATIC_FIELD_TAGS_WITHOUT_TRANSACTION_FIELDS,
- STATIC_SEMVER_TAGS,
- STATIC_SPAN_TAGS,
- } from 'sentry/components/events/searchBarFieldConstants';
- import SelectControl from 'sentry/components/forms/controls/selectControl';
- import SelectField from 'sentry/components/forms/fields/selectField';
- import FormField from 'sentry/components/forms/formField';
- import IdBadge from 'sentry/components/idBadge';
- import ListItem from 'sentry/components/list/listItem';
- import {MetricSearchBar} from 'sentry/components/metrics/metricSearchBar';
- import {normalizeDateTimeParams} from 'sentry/components/organizations/pageFilters/parse';
- import Panel from 'sentry/components/panels/panel';
- import PanelBody from 'sentry/components/panels/panelBody';
- import {SearchQueryBuilder} from 'sentry/components/searchQueryBuilder';
- import {InvalidReason} from 'sentry/components/searchSyntax/parser';
- import {SearchInvalidTag} from 'sentry/components/smartSearchBar/searchInvalidTag';
- 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 {Organization} from 'sentry/types/organization';
- import type {Environment, Project} from 'sentry/types/project';
- import {defined} from 'sentry/utils';
- import {isAggregateField, isMeasurement} from 'sentry/utils/discover/fields';
- import {getDisplayName} from 'sentry/utils/environment';
- import {DEVICE_CLASS_TAG_VALUES, FieldKind, isDeviceClass} from 'sentry/utils/fields';
- import {
- getMeasurements,
- type MeasurementCollection,
- } from 'sentry/utils/measurements/measurements';
- import {hasCustomMetrics} from 'sentry/utils/metrics/features';
- import {getMRI} from 'sentry/utils/metrics/mri';
- import {getOnDemandKeys, isOnDemandQueryString} from 'sentry/utils/onDemandMetrics';
- import {hasOnDemandMetricAlertFeature} from 'sentry/utils/onDemandMetrics/features';
- import withApi from 'sentry/utils/withApi';
- import withProjects from 'sentry/utils/withProjects';
- import withTags from 'sentry/utils/withTags';
- import WizardField from 'sentry/views/alerts/rules/metric/wizardField';
- import {
- convertDatasetEventTypesToSource,
- DATA_SOURCE_LABELS,
- DATA_SOURCE_TO_SET_AND_EVENT_TYPES,
- } from 'sentry/views/alerts/utils';
- import type {AlertType} from 'sentry/views/alerts/wizard/options';
- import {getSupportedAndOmittedTags} from 'sentry/views/alerts/wizard/options';
- import {getProjectOptions} from '../utils';
- import {isCrashFreeAlert} from './utils/isCrashFreeAlert';
- import {DEFAULT_AGGREGATE, DEFAULT_TRANSACTION_AGGREGATE} from './constants';
- import {AlertRuleComparisonType, Dataset, Datasource, TimeWindow} from './types';
- const TIME_WINDOW_MAP: Record<TimeWindow, string> = {
- [TimeWindow.ONE_MINUTE]: t('1 minute'),
- [TimeWindow.FIVE_MINUTES]: t('5 minutes'),
- [TimeWindow.TEN_MINUTES]: t('10 minutes'),
- [TimeWindow.FIFTEEN_MINUTES]: t('15 minutes'),
- [TimeWindow.THIRTY_MINUTES]: t('30 minutes'),
- [TimeWindow.ONE_HOUR]: t('1 hour'),
- [TimeWindow.TWO_HOURS]: t('2 hours'),
- [TimeWindow.FOUR_HOURS]: t('4 hours'),
- [TimeWindow.ONE_DAY]: t('24 hours'),
- };
- type Props = {
- aggregate: string;
- alertType: AlertType;
- api: Client;
- comparisonType: AlertRuleComparisonType;
- dataset: Dataset;
- disabled: boolean;
- 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;
- projects: Project[];
- router: InjectedRouter;
- tags: TagCollection;
- thresholdChart: React.ReactNode;
- timeWindow: number;
- // optional props
- activationCondition?: ActivationConditionType;
- allowChangeEventTypes?: boolean;
- comparisonDelta?: number;
- disableProjectSelector?: boolean;
- isErrorMigration?: boolean;
- isExtrapolatedChartData?: boolean;
- isForSpanMetric?: boolean;
- isTransactionMigration?: boolean;
- loadingProjects?: boolean;
- monitorType?: number;
- };
- type State = {
- environments: Environment[] | null;
- filterKeys: TagCollection;
- measurements: MeasurementCollection;
- };
- class RuleConditionsForm extends PureComponent<Props, State> {
- state: State = {
- environments: null,
- measurements: {},
- filterKeys: {},
- };
- componentDidMount() {
- this.fetchData();
- const measurements = getMeasurements();
- const filterKeys = this.getFilterKeys();
- this.setState({measurements, filterKeys});
- }
- componentDidUpdate(prevProps: Props) {
- if (prevProps.project.id === this.props.project.id) {
- return;
- }
- this.fetchData();
- }
- getFilterKeys = () => {
- const {organization, dataset, tags} = this.props;
- const {measurements} = this.state;
- const measurementsWithKind = Object.keys(measurements).reduce(
- (measurement_tags, key) => {
- measurement_tags[key] = {
- ...measurements[key],
- kind: FieldKind.MEASUREMENT,
- };
- return measurement_tags;
- },
- {}
- );
- const orgHasPerformanceView = organization.features.includes('performance-view');
- const combinedTags: TagCollection =
- dataset === Dataset.ERRORS
- ? Object.assign({}, STATIC_FIELD_TAGS_WITHOUT_TRANSACTION_FIELDS)
- : dataset === Dataset.TRANSACTIONS
- ? Object.assign(
- {},
- measurementsWithKind,
- STATIC_SPAN_TAGS,
- STATIC_FIELD_TAGS_WITHOUT_ERROR_FIELDS
- )
- : orgHasPerformanceView
- ? Object.assign({}, measurementsWithKind, STATIC_SPAN_TAGS, STATIC_FIELD_TAGS)
- : Object.assign({}, STATIC_FIELD_TAGS_WITHOUT_TRACING);
- const tagsWithKind = Object.keys(tags).reduce<Record<string, Tag>>((acc, key) => {
- acc[key] = {
- ...tags[key],
- kind: FieldKind.TAG,
- };
- return acc;
- }, {});
- const {omitTags} = getSupportedAndOmittedTags(dataset, organization);
- Object.assign(combinedTags, tagsWithKind, STATIC_SEMVER_TAGS);
- combinedTags.has = getHasTag(combinedTags);
- const list =
- omitTags && omitTags.length > 0 ? omit(combinedTags, omitTags) : combinedTags;
- return list;
- };
- formElemBaseStyle = {
- padding: `${space(0.5)}`,
- border: 'none',
- };
- async fetchData() {
- const {api, organization, project} = this.props;
- try {
- const environments = await api.requestPromise(
- `/projects/${organization.slug}/${project.slug}/environments/`,
- {
- query: {
- visibility: 'visible',
- },
- }
- );
- this.setState({environments});
- } catch (_err) {
- addErrorMessage(t('Unable to fetch environments'));
- }
- }
- getEventFieldValues = async (tag, query): Promise<string[]> => {
- const {api, organization, project, dataset, router} = this.props;
- if (isAggregateField(tag.key) || isMeasurement(tag.key)) {
- // We can't really auto suggest values for aggregate fields
- // or measurements, so we simply don't
- // NOTE: these in particular are for discover queries. We may not need/support these
- return Promise.resolve([]);
- }
- // device.class is stored as "numbers" in snuba, but we want to suggest high, medium,
- // and low search filter values because discover maps device.class to these values.
- if (isDeviceClass(tag.key)) {
- return Promise.resolve(DEVICE_CLASS_TAG_VALUES);
- }
- const values = await fetchTagValues({
- api,
- orgSlug: organization.slug,
- tagKey: tag.key,
- search: query,
- projectIds: [project.id],
- endpointParams: normalizeDateTimeParams(router.location.query), // allows searching for tags on transactions as well
- includeTransactions: true, // allows searching for tags on sessions as well
- includeSessions: dataset === Dataset.SESSIONS,
- });
- return values.filter(({name}) => defined(name)).map(({name}) => name);
- };
- get timeWindowOptions() {
- let options: Record<string, string> = TIME_WINDOW_MAP;
- const {alertType} = this.props;
- if (alertType === 'custom_metrics' || alertType === 'span_metrics') {
- // Do not show ONE MINUTE interval as an option for custom_metrics alert
- options = omit(options, TimeWindow.ONE_MINUTE.toString());
- }
- if (isCrashFreeAlert(this.props.dataset)) {
- options = pick(TIME_WINDOW_MAP, [
- // TimeWindow.THIRTY_MINUTES, leaving this option out until we figure out the sub-hour session resolution chart limitations
- TimeWindow.ONE_HOUR,
- TimeWindow.TWO_HOURS,
- TimeWindow.FOUR_HOURS,
- TimeWindow.ONE_DAY,
- ]);
- }
- if (this.props.comparisonType === AlertRuleComparisonType.DYNAMIC) {
- options = pick(TIME_WINDOW_MAP, [
- TimeWindow.FIFTEEN_MINUTES,
- TimeWindow.THIRTY_MINUTES,
- TimeWindow.ONE_HOUR,
- ]);
- }
- return Object.entries(options).map(([value, label]) => ({
- value: parseInt(value, 10),
- label: tct('[timeWindow] interval', {
- timeWindow: label.slice(-1) === 's' ? label.slice(0, -1) : label,
- }),
- }));
- }
- get searchPlaceholder() {
- switch (this.props.dataset) {
- case Dataset.ERRORS:
- return t('Filter events by level, message, and other properties\u2026');
- case Dataset.METRICS:
- case Dataset.SESSIONS:
- return t('Filter sessions by release version\u2026');
- default:
- return t('Filter transactions by URL, tags, and other properties\u2026');
- }
- }
- get selectControlStyles() {
- return {
- control: (provided: {[x: string]: string | number | boolean}) => ({
- ...provided,
- minWidth: 200,
- maxWidth: 300,
- }),
- container: (provided: {[x: string]: string | number | boolean}) => ({
- ...provided,
- margin: `${space(0.5)}`,
- }),
- };
- }
- renderEventTypeFilter() {
- const {organization, disabled, alertType, isErrorMigration} = this.props;
- const dataSourceOptions = [
- {
- label: t('Errors'),
- options: [
- {
- value: Datasource.ERROR_DEFAULT,
- label: DATA_SOURCE_LABELS[Datasource.ERROR_DEFAULT],
- },
- {
- value: Datasource.DEFAULT,
- label: DATA_SOURCE_LABELS[Datasource.DEFAULT],
- },
- {
- value: Datasource.ERROR,
- label: DATA_SOURCE_LABELS[Datasource.ERROR],
- },
- ],
- },
- ];
- if (
- organization.features.includes('performance-view') &&
- (alertType === 'custom_transactions' || alertType === 'custom_metrics')
- ) {
- dataSourceOptions.push({
- label: t('Transactions'),
- options: [
- {
- value: Datasource.TRANSACTION,
- label: DATA_SOURCE_LABELS[Datasource.TRANSACTION],
- },
- ],
- });
- }
- return (
- <FormField
- name="datasource"
- inline={false}
- style={{
- ...this.formElemBaseStyle,
- minWidth: 300,
- flex: 2,
- }}
- flexibleControlStateSize
- >
- {({onChange, onBlur, model}) => {
- const formDataset = model.getValue('dataset');
- const formEventTypes = model.getValue('eventTypes');
- const aggregate = model.getValue('aggregate');
- const mappedValue = convertDatasetEventTypesToSource(
- formDataset,
- formEventTypes
- );
- return (
- <SelectControl
- value={mappedValue}
- inFieldLabel={t('Events: ')}
- onChange={({value}) => {
- onChange(value, {});
- onBlur(value, {});
- // Reset the aggregate to the default (which works across
- // datatypes), otherwise we may send snuba an invalid query
- // (transaction aggregate on events datasource = bad).
- const newAggregate =
- value === Datasource.TRANSACTION
- ? DEFAULT_TRANSACTION_AGGREGATE
- : DEFAULT_AGGREGATE;
- if (alertType === 'custom_transactions' && aggregate !== newAggregate) {
- model.setValue('aggregate', newAggregate);
- }
- // set the value of the dataset and event type from data source
- const {dataset: datasetFromDataSource, eventTypes} =
- DATA_SOURCE_TO_SET_AND_EVENT_TYPES[value] ?? {};
- model.setValue('dataset', datasetFromDataSource);
- model.setValue('eventTypes', eventTypes);
- }}
- options={dataSourceOptions}
- isDisabled={disabled || isErrorMigration}
- />
- );
- }}
- </FormField>
- );
- }
- renderProjectSelector() {
- const {
- project: _selectedProject,
- projects, // note: org projects
- disabled,
- organization,
- disableProjectSelector,
- } = this.props;
- const projectOptions = getProjectOptions({
- organization,
- projects,
- isFormDisabled: disabled,
- });
- return (
- <FormField
- name="projectId"
- inline={false}
- style={{
- ...this.formElemBaseStyle,
- minWidth: 300,
- flex: 2,
- }}
- flexibleControlStateSize
- >
- {({onChange, onBlur, model}) => {
- const selectedProject =
- projects.find(({id}) => id === model.getValue('projectId')) ||
- _selectedProject;
- return (
- <SelectControl
- isDisabled={disabled || disableProjectSelector}
- value={selectedProject.id}
- 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
- ) {
- model.setValue('owner', `team:${nextSelectedProject.teams[0].id}`);
- }
- onChange(value, {});
- onBlur(value, {});
- }}
- components={{
- SingleValue: containerProps => (
- <components.ValueContainer {...containerProps}>
- <IdBadge
- project={selectedProject}
- avatarProps={{consistentWidth: true}}
- avatarSize={18}
- disableLink
- />
- </components.ValueContainer>
- ),
- }}
- />
- );
- }}
- </FormField>
- );
- }
- renderInterval() {
- const {
- organization,
- disabled,
- alertType,
- timeWindow,
- onTimeWindowChange,
- project,
- monitorType,
- isForSpanMetric,
- } = this.props;
- return (
- <Fragment>
- <StyledListItem>
- <StyledListTitle>
- <div>{t('Define your metric')}</div>
- </StyledListTitle>
- </StyledListItem>
- <FormRow>
- {isForSpanMetric ? null : (
- <WizardField
- name="aggregate"
- help={null}
- organization={organization}
- disabled={disabled}
- project={project}
- style={{
- ...this.formElemBaseStyle,
- flex: 1,
- }}
- inline={false}
- flexibleControlStateSize
- columnWidth={200}
- alertType={alertType}
- required
- />
- )}
- {monitorType !== MonitorType.ACTIVATED && (
- <SelectControl
- name="timeWindow"
- styles={this.selectControlStyles}
- options={this.timeWindowOptions}
- required={monitorType === MonitorType.CONTINUOUS}
- isDisabled={disabled}
- value={timeWindow}
- onChange={({value}) => onTimeWindowChange(value)}
- inline={false}
- flexibleControlStateSize
- />
- )}
- </FormRow>
- </Fragment>
- );
- }
- renderMonitorTypeSelect() {
- // TODO: disable select on edit
- const {
- activationCondition,
- isEditing,
- monitorType,
- onMonitorTypeSelect,
- onTimeWindowChange,
- timeWindow,
- } = this.props;
- return (
- <Fragment>
- <StyledListItem>
- <StyledListTitle>
- <div>{t('Select Monitor Type')}</div>
- </StyledListTitle>
- </StyledListItem>
- <FormRow>
- <MonitorSelect>
- <MonitorCard
- disabled={isEditing}
- position="left"
- isSelected={monitorType === MonitorType.CONTINUOUS}
- onClick={() =>
- isEditing
- ? null
- : onMonitorTypeSelect({
- monitorType: MonitorType.CONTINUOUS,
- })
- }
- >
- <strong>{t('Continuous')}</strong>
- <div>{t('Continuously monitor trends for the metrics outlined below')}</div>
- </MonitorCard>
- <MonitorCard
- disabled={isEditing}
- position="right"
- isSelected={monitorType === MonitorType.ACTIVATED}
- onClick={() =>
- isEditing
- ? null
- : onMonitorTypeSelect({
- monitorType: MonitorType.ACTIVATED,
- })
- }
- >
- <strong>Conditional</strong>
- {monitorType === MonitorType.ACTIVATED ? (
- <ActivatedAlertFields>
- {`${t('Monitor')} `}
- <SelectControl
- name="activationCondition"
- styles={this.selectControlStyles}
- disabled={isEditing}
- options={[
- {
- value: ActivationConditionType.RELEASE_CREATION,
- label: t('New Release'),
- },
- {
- value: ActivationConditionType.DEPLOY_CREATION,
- label: t('New Deploy'),
- },
- ]}
- required
- value={activationCondition}
- onChange={({value}) =>
- onMonitorTypeSelect({activationCondition: value})
- }
- inline={false}
- flexibleControlStateSize
- size="xs"
- />
- {` ${t('for')} `}
- <SelectControl
- name="timeWindow"
- styles={this.selectControlStyles}
- options={this.timeWindowOptions}
- value={timeWindow}
- onChange={({value}) => onTimeWindowChange(value)}
- inline={false}
- flexibleControlStateSize
- size="xs"
- />
- </ActivatedAlertFields>
- ) : (
- <div>
- {t('Temporarily monitor specified query given activation condition')}
- </div>
- )}
- </MonitorCard>
- </MonitorSelect>
- </FormRow>
- </Fragment>
- );
- }
- render() {
- const {
- alertType,
- organization,
- disabled,
- onFilterSearch,
- allowChangeEventTypes,
- dataset,
- isExtrapolatedChartData,
- isTransactionMigration,
- isErrorMigration,
- aggregate,
- project,
- } = this.props;
- const {environments, filterKeys} = this.state;
- const hasActivatedAlerts = organization.features.includes('activated-alert-rules');
- const environmentOptions: SelectValue<string | null>[] = [
- {
- value: null,
- label: t('All Environments'),
- },
- ...(environments?.map(env => ({value: env.name, label: getDisplayName(env)})) ??
- []),
- ];
- return (
- <Fragment>
- <ChartPanel>
- <StyledPanelBody>{this.props.thresholdChart}</StyledPanelBody>
- </ChartPanel>
- {isTransactionMigration ? (
- <Fragment>
- <Spacer />
- <HiddenListItem />
- <HiddenListItem />
- </Fragment>
- ) : (
- <Fragment>
- {isExtrapolatedChartData && (
- <OnDemandMetricAlert
- message={t(
- 'The chart data above is an estimate based on the stored transactions that match the filters specified.'
- )}
- />
- )}
- {hasActivatedAlerts && this.renderMonitorTypeSelect()}
- {!isErrorMigration && this.renderInterval()}
- <StyledListItem>{t('Filter events')}</StyledListItem>
- <FormRow noMargin columns={1 + (allowChangeEventTypes ? 1 : 0) + 1}>
- {this.renderProjectSelector()}
- <SelectField
- name="environment"
- placeholder={t('All Environments')}
- style={{
- ...this.formElemBaseStyle,
- minWidth: 230,
- flex: 1,
- }}
- styles={{
- singleValue: (base: any) => ({
- ...base,
- }),
- option: (base: any) => ({
- ...base,
- }),
- }}
- options={environmentOptions}
- isDisabled={
- disabled || this.state.environments === null || isErrorMigration
- }
- isClearable
- inline={false}
- flexibleControlStateSize
- />
- {allowChangeEventTypes && this.renderEventTypeFilter()}
- </FormRow>
- <FormRow>
- <FormField
- name="query"
- inline={false}
- style={{
- ...this.formElemBaseStyle,
- flex: '6 0 500px',
- }}
- flexibleControlStateSize
- >
- {({onChange, onBlur, onKeyDown, initialData, value}) => {
- return hasCustomMetrics(organization) &&
- alertType === 'custom_metrics' ? (
- <MetricSearchBar
- mri={getMRI(aggregate)}
- projectIds={[project.id]}
- placeholder={this.searchPlaceholder}
- query={initialData.query}
- defaultQuery={initialData?.query ?? ''}
- useFormWrapper={false}
- searchSource="alert_builder"
- onChange={query => {
- onFilterSearch(query, true);
- onChange(query, {});
- }}
- />
- ) : (
- <SearchContainer>
- {organization.features.includes('search-query-builder-alerts') ? (
- <SearchQueryBuilder
- initialQuery={initialData?.query ?? ''}
- getTagValues={this.getEventFieldValues}
- placeholder={this.searchPlaceholder}
- searchSource="alert_builder"
- filterKeys={filterKeys}
- disabled={disabled || isErrorMigration}
- onChange={onChange}
- invalidMessages={{
- [InvalidReason.WILDCARD_NOT_ALLOWED]: t(
- 'The wildcard operator is not supported here.'
- ),
- [InvalidReason.FREE_TEXT_NOT_ALLOWED]: t(
- 'Free text search is not allowed. If you want to partially match transaction names, use glob patterns like "transaction:*transaction-name*"'
- ),
- }}
- onSearch={query => {
- onFilterSearch(query, true);
- onChange(query, {});
- }}
- onBlur={(query, {parsedQuery}) => {
- onFilterSearch(query, parsedQuery);
- onBlur(query);
- }}
- // We only need strict validation for Transaction queries, everything else is fine
- disallowUnsupportedFilters={
- organization.features.includes('alert-allow-indexed') ||
- (hasOnDemandMetricAlertFeature(organization) &&
- isOnDemandQueryString(value))
- ? false
- : dataset === Dataset.GENERIC_METRICS
- }
- />
- ) : (
- <StyledSearchBar
- disallowWildcard={dataset === Dataset.SESSIONS}
- disallowFreeText={[
- Dataset.GENERIC_METRICS,
- Dataset.TRANSACTIONS,
- ].includes(dataset)}
- invalidMessages={{
- [InvalidReason.WILDCARD_NOT_ALLOWED]: t(
- 'The wildcard operator is not supported here.'
- ),
- [InvalidReason.FREE_TEXT_NOT_ALLOWED]: t(
- 'Free text search is not allowed. If you want to partially match transaction names, use glob patterns like "transaction:*transaction-name*"'
- ),
- }}
- customInvalidTagMessage={item => {
- if (dataset !== Dataset.GENERIC_METRICS) {
- return null;
- }
- return (
- <SearchInvalidTag
- message={tct(
- "The field [field] isn't supported for performance alerts.",
- {
- field: <code>{item.desc}</code>,
- }
- )}
- docLink="https://docs.sentry.io/product/alerts/create-alerts/metric-alert-config/#tags--properties"
- />
- );
- }}
- searchSource="alert_builder"
- defaultQuery={initialData?.query ?? ''}
- {...getSupportedAndOmittedTags(dataset, organization)}
- includeSessionTagsValues={dataset === Dataset.SESSIONS}
- disabled={disabled || isErrorMigration}
- useFormWrapper={false}
- organization={organization}
- placeholder={this.searchPlaceholder}
- onChange={onChange}
- query={initialData.query}
- // We only need strict validation for Transaction queries, everything else is fine
- highlightUnsupportedTags={
- organization.features.includes('alert-allow-indexed') ||
- (hasOnDemandMetricAlertFeature(organization) &&
- isOnDemandQueryString(value))
- ? false
- : dataset === Dataset.GENERIC_METRICS
- }
- onKeyDown={e => {
- /**
- * Do not allow enter key to submit the alerts form since it is unlikely
- * users will be ready to create the rule as this sits above required fields.
- */
- if (e.key === 'Enter') {
- e.preventDefault();
- e.stopPropagation();
- }
- onKeyDown?.(e);
- }}
- onClose={(query, {validSearch}) => {
- onFilterSearch(query, validSearch);
- onBlur(query);
- }}
- onSearch={query => {
- onFilterSearch(query, true);
- onChange(query, {});
- }}
- hasRecentSearches={dataset !== Dataset.SESSIONS}
- />
- )}
- {isExtrapolatedChartData && isOnDemandQueryString(value) && (
- <OnDemandWarningIcon
- color="gray500"
- msg={tct(
- `We don’t routinely collect metrics from [fields]. However, we’ll do so [strong:once this alert has been saved.]`,
- {
- fields: (
- <strong>
- {getOnDemandKeys(value)
- .map(key => `"${key}"`)
- .join(', ')}
- </strong>
- ),
- strong: <strong />,
- }
- )}
- />
- )}
- </SearchContainer>
- );
- }}
- </FormField>
- </FormRow>
- </Fragment>
- )}
- </Fragment>
- );
- }
- }
- const StyledListTitle = styled('div')`
- display: flex;
- span {
- margin-left: ${space(1)};
- }
- `;
- // This is a temporary hacky solution to hide list items without changing the numbering of the rest of the list
- // TODO(issues): Remove this once the migration is complete
- const HiddenListItem = styled(ListItem)`
- position: absolute;
- width: 0px;
- height: 0px;
- opacity: 0;
- pointer-events: none;
- `;
- const Spacer = styled('div')`
- margin-bottom: ${space(2)};
- `;
- const ChartPanel = styled(Panel)`
- margin-bottom: ${space(1)};
- `;
- const StyledPanelBody = styled(PanelBody)`
- ol,
- h4 {
- margin-bottom: ${space(1)};
- }
- `;
- const SearchContainer = styled('div')`
- display: flex;
- align-items: center;
- gap: ${space(1)};
- `;
- const StyledSearchBar = styled(SearchBar)`
- flex-grow: 1;
- ${p =>
- p.disabled &&
- `
- background: ${p.theme.backgroundSecondary};
- color: ${p.theme.disabled};
- cursor: not-allowed;
- `}
- `;
- const StyledListItem = styled(ListItem)`
- margin-bottom: ${space(0.5)};
- font-size: ${p => p.theme.fontSizeExtraLarge};
- line-height: 1.3;
- `;
- const FormRow = styled('div')<{columns?: number; noMargin?: boolean}>`
- display: flex;
- flex-direction: row;
- align-items: center;
- flex-wrap: wrap;
- margin-bottom: ${p => (p.noMargin ? 0 : space(4))};
- ${p =>
- p.columns !== undefined &&
- css`
- display: grid;
- grid-template-columns: repeat(${p.columns}, auto);
- `}
- `;
- 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')<MonitorCardProps>`
- 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)));
|