You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
180 lines
5.8 KiB
180 lines
5.8 KiB
import React, { useCallback, useEffect, useState } from 'react';
|
|
import { useDispatch } from 'react-redux';
|
|
import { css } from '@emotion/css';
|
|
import { GrafanaTheme2 } from '@grafana/data';
|
|
import { Button, ConfirmModal, HorizontalGroup, Icon, Tooltip, useStyles2 } from '@grafana/ui';
|
|
import EmptyListCTA from 'app/core/components/EmptyListCTA/EmptyListCTA';
|
|
import { AddAlertManagerModal } from './AddAlertManagerModal';
|
|
import {
|
|
addExternalAlertmanagersAction,
|
|
fetchExternalAlertmanagersAction,
|
|
fetchExternalAlertmanagersConfigAction,
|
|
} from '../../state/actions';
|
|
import { useExternalAmSelector } from '../../hooks/useExternalAmSelector';
|
|
|
|
export const ExternalAlertmanagers = () => {
|
|
const styles = useStyles2(getStyles);
|
|
const dispatch = useDispatch();
|
|
const [modalState, setModalState] = useState({ open: false, payload: [{ url: '' }] });
|
|
const [deleteModalState, setDeleteModalState] = useState({ open: false, index: 0 });
|
|
const externalAlertManagers = useExternalAmSelector();
|
|
|
|
useEffect(() => {
|
|
dispatch(fetchExternalAlertmanagersAction());
|
|
dispatch(fetchExternalAlertmanagersConfigAction());
|
|
const interval = setInterval(() => dispatch(fetchExternalAlertmanagersAction()), 5000);
|
|
|
|
return () => {
|
|
clearInterval(interval);
|
|
};
|
|
}, [dispatch]);
|
|
|
|
const onDelete = useCallback(
|
|
(index: number) => {
|
|
// to delete we need to filter the alertmanager from the list and repost
|
|
const newList = (externalAlertManagers ?? [])
|
|
.filter((am, i) => i !== index)
|
|
.map((am) => {
|
|
return am.url;
|
|
});
|
|
dispatch(addExternalAlertmanagersAction(newList));
|
|
setDeleteModalState({ open: false, index: 0 });
|
|
},
|
|
[externalAlertManagers, dispatch]
|
|
);
|
|
|
|
const onEdit = useCallback(() => {
|
|
const ams = externalAlertManagers ? [...externalAlertManagers] : [{ url: '' }];
|
|
setModalState((state) => ({
|
|
...state,
|
|
open: true,
|
|
payload: ams,
|
|
}));
|
|
}, [setModalState, externalAlertManagers]);
|
|
|
|
const onOpenModal = useCallback(() => {
|
|
setModalState((state) => {
|
|
const ams = externalAlertManagers ? [...externalAlertManagers, { url: '' }] : [{ url: '' }];
|
|
return {
|
|
...state,
|
|
open: true,
|
|
payload: ams,
|
|
};
|
|
});
|
|
}, [externalAlertManagers]);
|
|
|
|
const onCloseModal = useCallback(() => {
|
|
setModalState((state) => ({
|
|
...state,
|
|
open: false,
|
|
}));
|
|
}, [setModalState]);
|
|
|
|
const getStatusColor = (status: string) => {
|
|
switch (status) {
|
|
case 'active':
|
|
return 'green';
|
|
|
|
case 'pending':
|
|
return 'yellow';
|
|
|
|
default:
|
|
return 'red';
|
|
}
|
|
};
|
|
|
|
const noAlertmanagers = externalAlertManagers?.length === 0;
|
|
|
|
return (
|
|
<div>
|
|
<h4>External Alertmanagers</h4>
|
|
<div className={styles.muted}>
|
|
You can have your Grafana managed alerts be delivered to one or many external Alertmanager(s) in addition to the
|
|
internal Alertmanager by specifying their URLs below.
|
|
</div>
|
|
<div className={styles.actions}>
|
|
{!noAlertmanagers && (
|
|
<Button type="button" onClick={onOpenModal}>
|
|
Add Alertmanager
|
|
</Button>
|
|
)}
|
|
</div>
|
|
{noAlertmanagers ? (
|
|
<EmptyListCTA
|
|
title="You have not added any external alertmanagers"
|
|
onClick={onOpenModal}
|
|
buttonTitle="Add Alertmanager"
|
|
buttonIcon="bell-slash"
|
|
/>
|
|
) : (
|
|
<table className="filter-table form-inline filter-table--hover">
|
|
<thead>
|
|
<tr>
|
|
<th>Url</th>
|
|
<th>Status</th>
|
|
<th style={{ width: '2%' }}>Action</th>
|
|
</tr>
|
|
</thead>
|
|
<tbody>
|
|
{externalAlertManagers?.map((am, index) => {
|
|
return (
|
|
<tr key={index}>
|
|
<td>
|
|
<span className={styles.url}>{am.url}</span>
|
|
{am.actualUrl ? (
|
|
<Tooltip content={`Discovered ${am.actualUrl} from ${am.url}`} theme="info">
|
|
<Icon name="info-circle" />
|
|
</Tooltip>
|
|
) : null}
|
|
</td>
|
|
<td>
|
|
<Icon name="heart" style={{ color: getStatusColor(am.status) }} title={am.status} />
|
|
</td>
|
|
<td>
|
|
<HorizontalGroup>
|
|
<Button variant="secondary" type="button" onClick={onEdit} aria-label="Edit alertmanager">
|
|
<Icon name="pen" />
|
|
</Button>
|
|
<Button
|
|
variant="destructive"
|
|
aria-label="Remove alertmanager"
|
|
type="button"
|
|
onClick={() => setDeleteModalState({ open: true, index })}
|
|
>
|
|
<Icon name="trash-alt" />
|
|
</Button>
|
|
</HorizontalGroup>
|
|
</td>
|
|
</tr>
|
|
);
|
|
})}
|
|
</tbody>
|
|
</table>
|
|
)}
|
|
<ConfirmModal
|
|
isOpen={deleteModalState.open}
|
|
title="Remove Alertmanager"
|
|
body="Are you sure you want to remove this Alertmanager"
|
|
confirmText="Remove"
|
|
onConfirm={() => onDelete(deleteModalState.index)}
|
|
onDismiss={() => setDeleteModalState({ open: false, index: 0 })}
|
|
/>
|
|
{modalState.open && <AddAlertManagerModal onClose={onCloseModal} alertmanagers={modalState.payload} />}
|
|
</div>
|
|
);
|
|
};
|
|
|
|
const getStyles = (theme: GrafanaTheme2) => ({
|
|
url: css`
|
|
margin-right: ${theme.spacing(1)};
|
|
`,
|
|
muted: css`
|
|
color: ${theme.colors.text.secondary};
|
|
`,
|
|
actions: css`
|
|
margin-top: ${theme.spacing(2)};
|
|
display: flex;
|
|
justify-content: flex-end;
|
|
`,
|
|
table: css``,
|
|
});
|
|
|