import {Fragment} from 'react';
import type {RouteComponentProps} from 'react-router';
import styled from '@emotion/styled';
import type {Location} from 'history';
import moment from 'moment-timezone';
import type {Client} from 'sentry/api';
import {Alert} from 'sentry/components/alert';
import {getInterval} from 'sentry/components/charts/utils';
import * as Layout from 'sentry/components/layouts/thirds';
import Link from 'sentry/components/links/link';
import Panel from 'sentry/components/panels/panel';
import PanelBody from 'sentry/components/panels/panelBody';
import Placeholder from 'sentry/components/placeholder';
import type {ChangeData} from 'sentry/components/timeRangeSelector';
import {TimeRangeSelector} from 'sentry/components/timeRangeSelector';
import {Tooltip} from 'sentry/components/tooltip';
import {t, tct} from 'sentry/locale';
import {space} from 'sentry/styles/space';
import {RuleActionsCategories} from 'sentry/types/alerts';
import type {Organization} from 'sentry/types/organization';
import type {Project} from 'sentry/types/project';
import {findExtractionRuleCondition} from 'sentry/utils/metrics/extractionRules';
import {formatMRIField, parseField} from 'sentry/utils/metrics/mri';
import {shouldShowOnDemandMetricAlertUI} from 'sentry/utils/onDemandMetrics/features';
import {ErrorMigrationWarning} from 'sentry/views/alerts/rules/metric/details/errorMigrationWarning';
import MetricHistory from 'sentry/views/alerts/rules/metric/details/metricHistory';
import type {MetricRule} from 'sentry/views/alerts/rules/metric/types';
import {Dataset, TimePeriod} from 'sentry/views/alerts/rules/metric/types';
import {extractEventTypeFilterFromRule} from 'sentry/views/alerts/rules/metric/utils/getEventTypeFilter';
import {getFormattedSpanMetricField} from 'sentry/views/alerts/rules/metric/utils/getFormattedSpanMetric';
import {isSpanMetricAlert} from 'sentry/views/alerts/rules/metric/utils/isSpanMetricAlert';
import {isOnDemandMetricAlert} from 'sentry/views/alerts/rules/metric/utils/onDemandMetricAlert';
import {getAlertRuleActionCategory} from 'sentry/views/alerts/rules/utils';
import type {Incident} from 'sentry/views/alerts/types';
import {AlertRuleStatus} from 'sentry/views/alerts/types';
import {alertDetailsLink} from 'sentry/views/alerts/utils';
import {useMetricsExtractionRules} from 'sentry/views/settings/projectMetrics/utils/useMetricsExtractionRules';
import {isCrashFreeAlert} from '../utils/isCrashFreeAlert';
import {isCustomMetricAlert} from '../utils/isCustomMetricAlert';
import type {TimePeriodType} from './constants';
import {
API_INTERVAL_POINTS_LIMIT,
SELECTOR_RELATIVE_PERIODS,
TIME_WINDOWS,
} from './constants';
import MetricChart from './metricChart';
import RelatedIssues from './relatedIssues';
import RelatedTransactions from './relatedTransactions';
import {MetricDetailsSidebar} from './sidebar';
interface MetricDetailsBodyProps extends RouteComponentProps<{}, {}> {
api: Client;
location: Location;
organization: Organization;
timePeriod: TimePeriodType;
incidents?: Incident[];
project?: Project;
rule?: MetricRule;
selectedIncident?: Incident | null;
}
export default function MetricDetailsBody({
api,
project,
rule,
incidents,
organization,
timePeriod,
selectedIncident,
location,
router,
}: MetricDetailsBodyProps) {
const {data: metricExtractionRules} = useMetricsExtractionRules(
{
orgId: organization.slug,
projectId: project?.slug,
},
{enabled: isSpanMetricAlert(rule?.aggregate)}
);
function getPeriodInterval() {
const startDate = moment.utc(timePeriod.start);
const endDate = moment.utc(timePeriod.end);
const timeWindow = rule?.timeWindow;
const startEndDifferenceMs = endDate.diff(startDate);
if (
timeWindow &&
(startEndDifferenceMs < API_INTERVAL_POINTS_LIMIT * timeWindow * 60 * 1000 ||
// Special case 7 days * 1m interval over the api limit
startEndDifferenceMs === TIME_WINDOWS[TimePeriod.SEVEN_DAYS])
) {
return `${timeWindow}m`;
}
return getInterval({start: timePeriod.start, end: timePeriod.end}, 'high');
}
function getFilter(): string[] | null {
if (!rule) {
return null;
}
const {aggregate, dataset, query} = rule;
if (isSpanMetricAlert(aggregate)) {
const mri = parseField(aggregate)!.mri;
const usedCondition = findExtractionRuleCondition(mri, metricExtractionRules || []);
const fullQuery = usedCondition?.value
? query
? `(${usedCondition.value}) AND (${query})`
: usedCondition.value
: query;
return fullQuery.trim().split(' ');
}
if (isCrashFreeAlert(dataset) || isCustomMetricAlert(aggregate)) {
return query.trim().split(' ');
}
const eventType = extractEventTypeFilterFromRule(rule);
return (query ? `(${eventType}) AND (${query.trim()})` : eventType).split(' ');
}
const handleTimePeriodChange = (datetime: ChangeData) => {
const {start, end, relative} = datetime;
if (start && end) {
return router.push({
...location,
query: {
start: moment(start).utc().format(),
end: moment(end).utc().format(),
},
});
}
return router.push({
...location,
query: {
period: relative,
},
});
};
if (!rule || !project) {
return (
);
}
const {dataset, aggregate, query} = rule;
const eventType = extractEventTypeFilterFromRule(rule);
const queryWithTypeFilter = (
query ? `(${query}) AND (${eventType})` : eventType
).trim();
const relativeOptions = {
...SELECTOR_RELATIVE_PERIODS,
...(rule.timeWindow > 1 ? {[TimePeriod.FOURTEEN_DAYS]: t('Last 14 days')} : {}),
};
const isSnoozed = rule.snooze;
const ruleActionCategory = getAlertRuleActionCategory(rule);
const showOnDemandMetricAlertUI =
isOnDemandMetricAlert(dataset, aggregate, query) &&
shouldShowOnDemandMetricAlertUI(organization);
let formattedAggregate = aggregate;
if (isCustomMetricAlert(aggregate)) {
formattedAggregate = formatMRIField(aggregate);
}
if (isSpanMetricAlert(aggregate)) {
formattedAggregate = getFormattedSpanMetricField(aggregate, metricExtractionRules);
}
return (
{selectedIncident?.alertRule.status === AlertRuleStatus.SNAPSHOT && (
{t('Alert Rule settings have been updated since this alert was triggered.')}
)}
{isSnoozed && (
{ruleActionCategory === RuleActionsCategories.NO_DEFAULT
? tct(
"[creator] muted this alert so these notifications won't be sent in the future.",
{creator: rule.snoozeCreatedBy}
)
: tct(
"[creator] muted this alert[forEveryone]so you won't get these notifications in the future.",
{
creator: rule.snoozeCreatedBy,
forEveryone: rule.snoozeForEveryone ? ' for everyone ' : ' ',
}
)}
)}
{selectedIncident && (
Remove filter on alert #{selectedIncident.identifier}
)}
{/* TODO: add activation start/stop into chart */}
{[Dataset.METRICS, Dataset.SESSIONS, Dataset.ERRORS].includes(dataset) && (
)}
{dataset === Dataset.TRANSACTIONS && (
)}
);
}
const DetailWrapper = styled('div')`
display: flex;
flex: 1;
@media (max-width: ${p => p.theme.breakpoints.small}) {
flex-direction: column-reverse;
}
`;
const StyledLayoutBody = styled(Layout.Body)`
flex-grow: 0;
padding-bottom: 0 !important;
@media (min-width: ${p => p.theme.breakpoints.medium}) {
grid-template-columns: auto;
}
`;
const StyledAlert = styled(Alert)`
margin: 0;
`;
const ActivityWrapper = styled('div')`
display: flex;
flex: 1;
flex-direction: column;
width: 100%;
`;
const ChartPanel = styled(Panel)`
margin-top: ${space(2)};
`;
const StyledSubHeader = styled('div')`
margin-bottom: ${space(2)};
display: flex;
align-items: center;
`;
const StyledTimeRangeSelector = styled(TimeRangeSelector)`
margin-right: ${space(1)};
`;