import styled from '@emotion/styled';
import type {Location} from 'history';
import pick from 'lodash/pick';
import AlertBadge from 'sentry/components/badge/alertBadge';
import {SectionHeading} from 'sentry/components/charts/styles';
import EmptyStateWarning from 'sentry/components/emptyStateWarning';
import Link from 'sentry/components/links/link';
import LoadingError from 'sentry/components/loadingError';
import Placeholder from 'sentry/components/placeholder';
import TimeSince from 'sentry/components/timeSince';
import {URL_PARAM} from 'sentry/constants/pageFilters';
import {IconCheckmark, IconExclamation, IconFire, IconOpen} from 'sentry/icons';
import {t} from 'sentry/locale';
import {space} from 'sentry/styles/space';
import type {Organization} from 'sentry/types/organization';
import {useApiQuery} from 'sentry/utils/queryClient';
import type {Incident} from 'sentry/views/alerts/types';
import {IncidentStatus} from 'sentry/views/alerts/types';
import MissingAlertsButtons from './missingFeatureButtons/missingAlertsButtons';
import {SectionHeadingLink, SectionHeadingWrapper, SidebarSection} from './styles';
const PLACEHOLDER_AND_EMPTY_HEIGHT = '172px';
interface AlertRowProps {
alert: Incident;
orgSlug: string;
}
function AlertRow({alert, orgSlug}: AlertRowProps) {
const {status, identifier, title, dateClosed, dateStarted} = alert;
const isResolved = status === IncidentStatus.CLOSED;
const isWarning = status === IncidentStatus.WARNING;
const Icon = isResolved ? IconCheckmark : isWarning ? IconExclamation : IconFire;
const statusProps = {isResolved, isWarning};
return (
{title}
{isResolved ? t('Resolved') : t('Triggered')}{' '}
{isResolved ? (
dateClosed ? (
) : null
) : (
)}
);
}
interface ProjectLatestAlertsProps {
isProjectStabilized: boolean;
location: Location;
organization: Organization;
projectSlug: string;
}
function ProjectLatestAlerts({
location,
organization,
isProjectStabilized,
projectSlug,
}: ProjectLatestAlertsProps) {
const query = {
...pick(location.query, Object.values(URL_PARAM)),
per_page: 3,
};
const {
data: unresolvedAlerts = [],
isLoading: unresolvedAlertsIsLoading,
isError: unresolvedAlertsIsError,
} = useApiQuery(
[
`/organizations/${organization.slug}/incidents/`,
{query: {...query, status: 'open'}},
],
{staleTime: 0, enabled: isProjectStabilized}
);
const {
data: resolvedAlerts = [],
isLoading: resolvedAlertsIsLoading,
isError: resolvedAlertsIsError,
} = useApiQuery(
[
`/organizations/${organization.slug}/incidents/`,
{query: {...query, status: 'closed'}},
],
{staleTime: 0, enabled: isProjectStabilized}
);
const alertsUnresolvedAndResolved = [...unresolvedAlerts, ...resolvedAlerts];
const shouldLoadAlertRules =
alertsUnresolvedAndResolved.length === 0 &&
!unresolvedAlertsIsLoading &&
!resolvedAlertsIsLoading;
// This is only used to determine if we should show the "Create Alert" button
const {data: alertRules = [], isLoading: alertRulesLoading} = useApiQuery(
[
`/organizations/${organization.slug}/alert-rules/`,
{
query: {
...pick(location.query, Object.values(URL_PARAM)),
// Sort by name
asc: 1,
per_page: 1,
},
},
],
{
staleTime: 0,
enabled: shouldLoadAlertRules,
}
);
function renderAlertRules() {
if (unresolvedAlertsIsError || resolvedAlertsIsError) {
return ;
}
const isLoading = unresolvedAlertsIsLoading || resolvedAlertsIsLoading;
if (isLoading || (shouldLoadAlertRules && alertRulesLoading)) {
return ;
}
const hasAlertRule = alertsUnresolvedAndResolved.length > 0 || alertRules?.length > 0;
if (!hasAlertRule) {
return (
);
}
if (alertsUnresolvedAndResolved.length === 0) {
return (
{t('No alerts found')}
);
}
return alertsUnresolvedAndResolved
.slice(0, 3)
.map(alert => (
));
}
return (
{t('Latest Alerts')}
{/* as this is a link to latest alerts, we want to only preserve project and environment */}
{renderAlertRules()}
);
}
const AlertRowLink = styled(Link)`
display: flex;
align-items: center;
height: 40px;
margin-bottom: ${space(3)};
margin-left: ${space(0.5)};
&,
&:hover,
&:focus {
color: inherit;
}
&:first-child {
margin-top: ${space(1)};
}
`;
type StatusColorProps = {
isResolved: boolean;
isWarning: boolean;
};
const getStatusColor = ({isResolved, isWarning}: StatusColorProps) =>
isResolved ? 'successText' : isWarning ? 'warningText' : 'errorText';
const AlertBadgeWrapper = styled('div')<{icon: typeof IconExclamation}>`
display: flex;
align-items: center;
justify-content: center;
flex-shrink: 0;
/* icon warning needs to be treated differently to look visually centered */
line-height: ${p => (p.icon === IconExclamation ? undefined : 1)};
`;
const AlertDetails = styled('div')`
font-size: ${p => p.theme.fontSizeMedium};
margin-left: ${space(1.5)};
${p => p.theme.overflowEllipsis}
line-height: 1.35;
`;
const AlertTitle = styled('div')`
font-weight: 400;
overflow: hidden;
text-overflow: ellipsis;
`;
const AlertDate = styled('span')`
color: ${p => p.theme[getStatusColor(p)]};
`;
const StyledEmptyStateWarning = styled(EmptyStateWarning)`
height: ${PLACEHOLDER_AND_EMPTY_HEIGHT};
justify-content: center;
`;
export default ProjectLatestAlerts;