|
@@ -0,0 +1,456 @@
|
|
|
+import {useMemo, useState} from 'react';
|
|
|
+import styled from '@emotion/styled';
|
|
|
+
|
|
|
+import Access from 'sentry/components/acl/access';
|
|
|
+import ActorAvatar from 'sentry/components/avatar/actorAvatar';
|
|
|
+import TeamAvatar from 'sentry/components/avatar/teamAvatar';
|
|
|
+import AlertBadge from 'sentry/components/badge/alertBadge';
|
|
|
+import {openConfirmModal} from 'sentry/components/confirm';
|
|
|
+import DropdownAutoComplete from 'sentry/components/dropdownAutoComplete';
|
|
|
+import type {ItemsBeforeFilter} from 'sentry/components/dropdownAutoComplete/types';
|
|
|
+import DropdownBubble from 'sentry/components/dropdownBubble';
|
|
|
+import type {MenuItemProps} from 'sentry/components/dropdownMenu';
|
|
|
+import {DropdownMenu} from 'sentry/components/dropdownMenu';
|
|
|
+import ErrorBoundary from 'sentry/components/errorBoundary';
|
|
|
+import IdBadge from 'sentry/components/idBadge';
|
|
|
+import LoadingIndicator from 'sentry/components/loadingIndicator';
|
|
|
+import TextOverflow from 'sentry/components/textOverflow';
|
|
|
+import TimeSince from 'sentry/components/timeSince';
|
|
|
+import {Tooltip} from 'sentry/components/tooltip';
|
|
|
+import {IconArrow, IconChevron, IconEllipsis, IconMute, IconUser} from 'sentry/icons';
|
|
|
+import {t, tct} from 'sentry/locale';
|
|
|
+import {space} from 'sentry/styles/space';
|
|
|
+import type {Actor, Project} from 'sentry/types';
|
|
|
+import type {ColorOrAlias} from 'sentry/utils/theme';
|
|
|
+import {useUserTeams} from 'sentry/utils/useUserTeams';
|
|
|
+import {getThresholdUnits} from 'sentry/views/alerts/rules/metric/constants';
|
|
|
+import {
|
|
|
+ AlertRuleComparisonType,
|
|
|
+ AlertRuleThresholdType,
|
|
|
+ AlertRuleTriggerType,
|
|
|
+} from 'sentry/views/alerts/rules/metric/types';
|
|
|
+
|
|
|
+import type {CombinedMetricIssueAlerts, MetricAlert} from '../../types';
|
|
|
+import {ActivationStatus, CombinedAlertType, IncidentStatus} from '../../types';
|
|
|
+
|
|
|
+type Props = {
|
|
|
+ hasEditAccess: boolean;
|
|
|
+ onDelete: (projectId: string, rule: CombinedMetricIssueAlerts) => void;
|
|
|
+ onOwnerChange: (
|
|
|
+ projectId: string,
|
|
|
+ rule: CombinedMetricIssueAlerts,
|
|
|
+ ownerValue: string
|
|
|
+ ) => void;
|
|
|
+ orgId: string;
|
|
|
+ projects: Project[];
|
|
|
+ projectsLoaded: boolean;
|
|
|
+ rule: MetricAlert;
|
|
|
+};
|
|
|
+
|
|
|
+function ActivatedRuleListRow({
|
|
|
+ rule,
|
|
|
+ projectsLoaded,
|
|
|
+ projects,
|
|
|
+ orgId,
|
|
|
+ onDelete,
|
|
|
+ onOwnerChange,
|
|
|
+ hasEditAccess,
|
|
|
+}: Props) {
|
|
|
+ const {teams: userTeams} = useUserTeams();
|
|
|
+ const [assignee, setAssignee] = useState<string>('');
|
|
|
+ const isWaiting = useMemo(
|
|
|
+ () =>
|
|
|
+ !rule.activations?.length ||
|
|
|
+ (rule.activations?.length && rule.activations[0].isComplete),
|
|
|
+ [rule]
|
|
|
+ );
|
|
|
+
|
|
|
+ function renderLatestActivation(): React.ReactNode {
|
|
|
+ if (!rule.activations?.length) {
|
|
|
+ return t('Alert has not been activated yet');
|
|
|
+ }
|
|
|
+
|
|
|
+ return (
|
|
|
+ <div>
|
|
|
+ {t('Last activated ')}
|
|
|
+ <TimeSince date={rule.activations[0].dateCreated} />
|
|
|
+ </div>
|
|
|
+ );
|
|
|
+ }
|
|
|
+
|
|
|
+ function renderSnoozeStatus(): React.ReactNode {
|
|
|
+ return (
|
|
|
+ <IssueAlertStatusWrapper>
|
|
|
+ <IconMute size="sm" color="subText" />
|
|
|
+ {t('Muted')}
|
|
|
+ </IssueAlertStatusWrapper>
|
|
|
+ );
|
|
|
+ }
|
|
|
+
|
|
|
+ function renderAlertRuleStatus(): React.ReactNode {
|
|
|
+ if (rule.snooze) {
|
|
|
+ return renderSnoozeStatus();
|
|
|
+ }
|
|
|
+
|
|
|
+ const isUnhealthy =
|
|
|
+ rule.latestIncident?.status !== undefined &&
|
|
|
+ [IncidentStatus.CRITICAL, IncidentStatus.WARNING].includes(
|
|
|
+ rule.latestIncident.status
|
|
|
+ );
|
|
|
+
|
|
|
+ let iconColor: ColorOrAlias = 'successText';
|
|
|
+ let iconDirection: 'up' | 'down' =
|
|
|
+ rule.thresholdType === AlertRuleThresholdType.ABOVE ? 'down' : 'up';
|
|
|
+ let thresholdTypeText =
|
|
|
+ rule.thresholdType === AlertRuleThresholdType.ABOVE ? t('Below') : t('Above');
|
|
|
+ if (isUnhealthy) {
|
|
|
+ iconColor =
|
|
|
+ rule.latestIncident?.status === IncidentStatus.CRITICAL
|
|
|
+ ? 'errorText'
|
|
|
+ : 'warningText';
|
|
|
+ // if unhealthy, swap icon direction
|
|
|
+ iconDirection = rule.thresholdType === AlertRuleThresholdType.ABOVE ? 'up' : 'down';
|
|
|
+ thresholdTypeText =
|
|
|
+ rule.thresholdType === AlertRuleThresholdType.ABOVE ? t('Above') : t('Below');
|
|
|
+ }
|
|
|
+
|
|
|
+ let threshold = rule.triggers.find(
|
|
|
+ ({label}) => label === AlertRuleTriggerType.CRITICAL
|
|
|
+ )?.alertThreshold;
|
|
|
+ if (isUnhealthy && rule.latestIncident?.status === IncidentStatus.WARNING) {
|
|
|
+ threshold = rule.triggers.find(
|
|
|
+ ({label}) => label === AlertRuleTriggerType.WARNING
|
|
|
+ )?.alertThreshold;
|
|
|
+ } else if (!isUnhealthy && rule.latestIncident && rule.resolveThreshold) {
|
|
|
+ threshold = rule.resolveThreshold;
|
|
|
+ }
|
|
|
+
|
|
|
+ return (
|
|
|
+ <FlexCenter>
|
|
|
+ <IconArrow color={iconColor} direction={iconDirection} />
|
|
|
+ <TriggerText>
|
|
|
+ {`${thresholdTypeText} ${threshold}`}
|
|
|
+ {getThresholdUnits(
|
|
|
+ rule.aggregate,
|
|
|
+ rule.comparisonDelta
|
|
|
+ ? AlertRuleComparisonType.CHANGE
|
|
|
+ : AlertRuleComparisonType.COUNT
|
|
|
+ )}
|
|
|
+ </TriggerText>
|
|
|
+ </FlexCenter>
|
|
|
+ );
|
|
|
+ }
|
|
|
+
|
|
|
+ const slug = rule.projects[0];
|
|
|
+ const editLink = `/organizations/${orgId}/alerts/metric-rules/${slug}/${rule.id}/`;
|
|
|
+
|
|
|
+ const duplicateLink = {
|
|
|
+ pathname: `/organizations/${orgId}/alerts/new/${
|
|
|
+ rule.type === CombinedAlertType.METRIC ? 'metric' : 'issue'
|
|
|
+ }/`,
|
|
|
+ query: {
|
|
|
+ project: slug,
|
|
|
+ duplicateRuleId: rule.id,
|
|
|
+ createFromDuplicate: true,
|
|
|
+ referrer: 'alert_stream',
|
|
|
+ },
|
|
|
+ };
|
|
|
+
|
|
|
+ const ownerId = rule.owner?.split(':')[1];
|
|
|
+ const teamActor = ownerId
|
|
|
+ ? {type: 'team' as Actor['type'], id: ownerId, name: ''}
|
|
|
+ : null;
|
|
|
+
|
|
|
+ const canEdit = ownerId ? userTeams.some(team => team.id === ownerId) : true;
|
|
|
+
|
|
|
+ const actions: MenuItemProps[] = [
|
|
|
+ {
|
|
|
+ key: 'edit',
|
|
|
+ label: t('Edit'),
|
|
|
+ to: editLink,
|
|
|
+ },
|
|
|
+ {
|
|
|
+ key: 'duplicate',
|
|
|
+ label: t('Duplicate'),
|
|
|
+ to: duplicateLink,
|
|
|
+ },
|
|
|
+ {
|
|
|
+ key: 'delete',
|
|
|
+ label: t('Delete'),
|
|
|
+ priority: 'danger',
|
|
|
+ onAction: () => {
|
|
|
+ openConfirmModal({
|
|
|
+ onConfirm: () => onDelete(slug, rule),
|
|
|
+ header: <h5>{t('Delete Alert Rule?')}</h5>,
|
|
|
+ message: t(
|
|
|
+ 'Are you sure you want to delete "%s"? You won\'t be able to view the history of this alert once it\'s deleted.',
|
|
|
+ rule.name
|
|
|
+ ),
|
|
|
+ confirmText: t('Delete Rule'),
|
|
|
+ priority: 'danger',
|
|
|
+ });
|
|
|
+ },
|
|
|
+ },
|
|
|
+ ];
|
|
|
+
|
|
|
+ function handleOwnerChange({value}: {value: string}) {
|
|
|
+ const ownerValue = value && `team:${value}`;
|
|
|
+ setAssignee(ownerValue);
|
|
|
+ onOwnerChange(slug, rule, ownerValue);
|
|
|
+ }
|
|
|
+
|
|
|
+ const unassignedOption: ItemsBeforeFilter[number] = {
|
|
|
+ value: '',
|
|
|
+ label: (
|
|
|
+ <MenuItemWrapper>
|
|
|
+ <PaddedIconUser size="lg" />
|
|
|
+ <Label>{t('Unassigned')}</Label>
|
|
|
+ </MenuItemWrapper>
|
|
|
+ ),
|
|
|
+ searchKey: 'unassigned',
|
|
|
+ actor: '',
|
|
|
+ disabled: false,
|
|
|
+ };
|
|
|
+
|
|
|
+ const project = projects.find(p => p.slug === slug);
|
|
|
+ const filteredProjectTeams = (project?.teams ?? []).filter(projTeam => {
|
|
|
+ return userTeams.some(team => team.id === projTeam.id);
|
|
|
+ });
|
|
|
+ const dropdownTeams = filteredProjectTeams
|
|
|
+ .map<ItemsBeforeFilter[number]>((team, idx) => ({
|
|
|
+ value: team.id,
|
|
|
+ searchKey: team.slug,
|
|
|
+ label: (
|
|
|
+ <MenuItemWrapper data-test-id="assignee-option" key={idx}>
|
|
|
+ <IconContainer>
|
|
|
+ <TeamAvatar team={team} size={24} />
|
|
|
+ </IconContainer>
|
|
|
+ <Label>#{team.slug}</Label>
|
|
|
+ </MenuItemWrapper>
|
|
|
+ ),
|
|
|
+ }))
|
|
|
+ .concat(unassignedOption);
|
|
|
+
|
|
|
+ const teamId = assignee?.split(':')[1];
|
|
|
+ const teamName = filteredProjectTeams.find(team => team.id === teamId);
|
|
|
+
|
|
|
+ const assigneeTeamActor = assignee && {
|
|
|
+ type: 'team' as Actor['type'],
|
|
|
+ id: teamId,
|
|
|
+ name: '',
|
|
|
+ };
|
|
|
+
|
|
|
+ const avatarElement = assigneeTeamActor ? (
|
|
|
+ <ActorAvatar
|
|
|
+ actor={assigneeTeamActor}
|
|
|
+ className="avatar"
|
|
|
+ size={24}
|
|
|
+ tooltipOptions={{overlayStyle: {textAlign: 'left'}}}
|
|
|
+ tooltip={tct('Assigned to [name]', {name: teamName && `#${teamName.name}`})}
|
|
|
+ />
|
|
|
+ ) : (
|
|
|
+ <Tooltip isHoverable skipWrapper title={t('Unassigned')}>
|
|
|
+ <PaddedIconUser size="lg" color="gray400" />
|
|
|
+ </Tooltip>
|
|
|
+ );
|
|
|
+
|
|
|
+ return (
|
|
|
+ <ErrorBoundary>
|
|
|
+ <AlertNameWrapper>
|
|
|
+ <AlertNameAndStatus>
|
|
|
+ <AlertName>{rule.name}</AlertName>
|
|
|
+ <AlertActivationDate>{renderLatestActivation()}</AlertActivationDate>
|
|
|
+ </AlertNameAndStatus>
|
|
|
+ </AlertNameWrapper>
|
|
|
+ <FlexCenter>
|
|
|
+ <FlexCenter>
|
|
|
+ <Tooltip
|
|
|
+ title={tct('Metric Alert Status: [status]', {
|
|
|
+ status: isWaiting ? 'Ready to monitor' : 'Monitoring',
|
|
|
+ })}
|
|
|
+ >
|
|
|
+ <AlertBadge
|
|
|
+ status={rule?.latestIncident?.status}
|
|
|
+ activationStatus={
|
|
|
+ isWaiting ? ActivationStatus.WAITING : ActivationStatus.MONITORING
|
|
|
+ }
|
|
|
+ />
|
|
|
+ </Tooltip>
|
|
|
+ </FlexCenter>
|
|
|
+ <MarginLeft>{renderAlertRuleStatus()}</MarginLeft>
|
|
|
+ </FlexCenter>
|
|
|
+ <FlexCenter>
|
|
|
+ <ProjectBadgeContainer>
|
|
|
+ <ProjectBadge
|
|
|
+ avatarSize={18}
|
|
|
+ project={projectsLoaded && project ? project : {slug}}
|
|
|
+ />
|
|
|
+ </ProjectBadgeContainer>
|
|
|
+ </FlexCenter>
|
|
|
+
|
|
|
+ <FlexCenter>
|
|
|
+ {teamActor ? (
|
|
|
+ <ActorAvatar actor={teamActor} size={24} />
|
|
|
+ ) : (
|
|
|
+ <AssigneeWrapper>
|
|
|
+ {!projectsLoaded && <StyledLoadingIndicator mini />}
|
|
|
+ {projectsLoaded && (
|
|
|
+ <DropdownAutoComplete
|
|
|
+ data-test-id="alert-row-assignee"
|
|
|
+ maxHeight={400}
|
|
|
+ onOpen={e => {
|
|
|
+ e?.stopPropagation();
|
|
|
+ }}
|
|
|
+ items={dropdownTeams}
|
|
|
+ alignMenu="right"
|
|
|
+ onSelect={handleOwnerChange}
|
|
|
+ itemSize="small"
|
|
|
+ searchPlaceholder={t('Filter teams')}
|
|
|
+ disableLabelPadding
|
|
|
+ emptyHidesInput
|
|
|
+ disabled={!hasEditAccess}
|
|
|
+ >
|
|
|
+ {({getActorProps, isOpen}) => (
|
|
|
+ <DropdownButton {...getActorProps({})}>
|
|
|
+ {avatarElement}
|
|
|
+ {hasEditAccess && (
|
|
|
+ <StyledChevron direction={isOpen ? 'up' : 'down'} size="xs" />
|
|
|
+ )}
|
|
|
+ </DropdownButton>
|
|
|
+ )}
|
|
|
+ </DropdownAutoComplete>
|
|
|
+ )}
|
|
|
+ </AssigneeWrapper>
|
|
|
+ )}
|
|
|
+ </FlexCenter>
|
|
|
+ <ActionsColumn>
|
|
|
+ <Access access={['alerts:write']}>
|
|
|
+ {({hasAccess}) => (
|
|
|
+ <DropdownMenu
|
|
|
+ items={actions}
|
|
|
+ position="bottom-end"
|
|
|
+ triggerProps={{
|
|
|
+ 'aria-label': t('Actions'),
|
|
|
+ size: 'xs',
|
|
|
+ icon: <IconEllipsis />,
|
|
|
+ showChevron: false,
|
|
|
+ }}
|
|
|
+ disabledKeys={hasAccess && canEdit ? [] : ['delete']}
|
|
|
+ />
|
|
|
+ )}
|
|
|
+ </Access>
|
|
|
+ </ActionsColumn>
|
|
|
+ </ErrorBoundary>
|
|
|
+ );
|
|
|
+}
|
|
|
+
|
|
|
+// TODO: see static/app/components/profiling/flex.tsx and utilize the FlexContainer styled component
|
|
|
+const FlexCenter = styled('div')`
|
|
|
+ display: flex;
|
|
|
+ align-items: center;
|
|
|
+`;
|
|
|
+
|
|
|
+const IssueAlertStatusWrapper = styled('div')`
|
|
|
+ display: flex;
|
|
|
+ align-items: center;
|
|
|
+ gap: ${space(1)};
|
|
|
+ line-height: 2;
|
|
|
+`;
|
|
|
+
|
|
|
+const AlertNameWrapper = styled('div')<{isIssueAlert?: boolean}>`
|
|
|
+ ${p => p.theme.overflowEllipsis}
|
|
|
+ display: flex;
|
|
|
+ align-items: center;
|
|
|
+ gap: ${space(2)};
|
|
|
+ ${p => p.isIssueAlert && `padding: ${space(3)} ${space(2)}; line-height: 2.4;`}
|
|
|
+`;
|
|
|
+
|
|
|
+const AlertNameAndStatus = styled('div')`
|
|
|
+ ${p => p.theme.overflowEllipsis}
|
|
|
+ line-height: 1.35;
|
|
|
+`;
|
|
|
+
|
|
|
+const AlertName = styled('div')`
|
|
|
+ ${p => p.theme.overflowEllipsis}
|
|
|
+ font-size: ${p => p.theme.fontSizeLarge};
|
|
|
+`;
|
|
|
+
|
|
|
+const AlertActivationDate = styled('div')`
|
|
|
+ color: ${p => p.theme.gray300};
|
|
|
+`;
|
|
|
+
|
|
|
+const ProjectBadgeContainer = styled('div')`
|
|
|
+ width: 100%;
|
|
|
+`;
|
|
|
+
|
|
|
+const ProjectBadge = styled(IdBadge)`
|
|
|
+ flex-shrink: 0;
|
|
|
+`;
|
|
|
+
|
|
|
+const TriggerText = styled('div')`
|
|
|
+ margin-left: ${space(1)};
|
|
|
+ white-space: nowrap;
|
|
|
+ font-variant-numeric: tabular-nums;
|
|
|
+`;
|
|
|
+
|
|
|
+const ActionsColumn = styled('div')`
|
|
|
+ display: flex;
|
|
|
+ align-items: center;
|
|
|
+ justify-content: center;
|
|
|
+ padding: ${space(1)};
|
|
|
+`;
|
|
|
+
|
|
|
+const AssigneeWrapper = styled('div')`
|
|
|
+ display: flex;
|
|
|
+ justify-content: flex-end;
|
|
|
+
|
|
|
+ /* manually align menu underneath dropdown caret */
|
|
|
+ ${DropdownBubble} {
|
|
|
+ right: -14px;
|
|
|
+ }
|
|
|
+`;
|
|
|
+
|
|
|
+const DropdownButton = styled('div')`
|
|
|
+ display: flex;
|
|
|
+ align-items: center;
|
|
|
+ font-size: 20px;
|
|
|
+`;
|
|
|
+
|
|
|
+const StyledChevron = styled(IconChevron)`
|
|
|
+ margin-left: ${space(1)};
|
|
|
+`;
|
|
|
+
|
|
|
+const PaddedIconUser = styled(IconUser)`
|
|
|
+ padding: ${space(0.25)};
|
|
|
+`;
|
|
|
+
|
|
|
+const IconContainer = styled('div')`
|
|
|
+ display: flex;
|
|
|
+ align-items: center;
|
|
|
+ justify-content: center;
|
|
|
+ width: ${p => p.theme.iconSizes.lg};
|
|
|
+ height: ${p => p.theme.iconSizes.lg};
|
|
|
+ flex-shrink: 0;
|
|
|
+`;
|
|
|
+
|
|
|
+const MenuItemWrapper = styled('div')`
|
|
|
+ display: flex;
|
|
|
+ align-items: center;
|
|
|
+ font-size: ${p => p.theme.fontSizeSmall};
|
|
|
+`;
|
|
|
+
|
|
|
+const Label = styled(TextOverflow)`
|
|
|
+ margin-left: ${space(0.75)};
|
|
|
+`;
|
|
|
+
|
|
|
+const MarginLeft = styled('div')`
|
|
|
+ margin-left: ${space(1)};
|
|
|
+`;
|
|
|
+
|
|
|
+const StyledLoadingIndicator = styled(LoadingIndicator)`
|
|
|
+ height: 24px;
|
|
|
+ margin: 0;
|
|
|
+ margin-right: ${space(1.5)};
|
|
|
+`;
|
|
|
+
|
|
|
+export default ActivatedRuleListRow;
|