188 lines
5.9 KiB
TypeScript
188 lines
5.9 KiB
TypeScript
import React, { PureComponent } from 'react';
|
|
import ReactTable from 'react-table';
|
|
import * as utils from '../../datasource-zabbix/utils';
|
|
import { ProblemsPanelOptions, Trigger, ZBXEvent, GFTimeRange, RTCell } from '../types';
|
|
import EventTag from './EventTag';
|
|
import ProblemDetails from './ProblemDetails';
|
|
import { AckProblemData } from './Modal';
|
|
|
|
export interface ProblemListProps {
|
|
problems: Trigger[];
|
|
panelOptions: ProblemsPanelOptions;
|
|
loading?: boolean;
|
|
timeRange?: GFTimeRange;
|
|
getProblemEvents: (ids: string[]) => ZBXEvent[];
|
|
onProblemAck: (problem: Trigger, data: AckProblemData) => void;
|
|
}
|
|
|
|
interface ProblemListState {
|
|
expanded: any;
|
|
page: number;
|
|
}
|
|
|
|
export class ProblemList extends PureComponent<ProblemListProps, ProblemListState> {
|
|
rootWidth: number;
|
|
rootRef: any;
|
|
|
|
constructor(props) {
|
|
super(props);
|
|
this.state = {
|
|
expanded: {},
|
|
page: 0,
|
|
};
|
|
}
|
|
|
|
setRootRef = ref => {
|
|
this.rootRef = ref;
|
|
}
|
|
|
|
handleProblemAck = (problem: Trigger, data: AckProblemData) => {
|
|
return this.props.onProblemAck(problem, data);
|
|
}
|
|
|
|
buildColumns() {
|
|
const result = [];
|
|
const options = this.props.panelOptions;
|
|
const problems = this.props.problems;
|
|
const timeColWidth = problems && problems.length ? problems[0].lastchange.length * 9 : 160;
|
|
const highlightNewerThan = options.highlightNewEvents && options.highlightNewerThan;
|
|
const statusCell = props => StatusCell(props, options.okEventColor, DEFAULT_PROBLEM_COLOR, highlightNewerThan);
|
|
const columns = [
|
|
{ Header: 'Host', accessor: 'host', show: options.hostField },
|
|
{ Header: 'Host (Technical Name)', accessor: 'hostTechName', show: options.hostTechNameField },
|
|
{ Header: 'Host Groups', accessor: 'groups', show: options.hostGroups, Cell: GroupCell },
|
|
{ Header: 'Proxy', accessor: 'proxy', show: options.hostProxy },
|
|
{ Header: 'Severity', show: options.severityField, className: 'problem-severity', width: 120,
|
|
accessor: problem => problem.priority,
|
|
id: 'severity',
|
|
Cell: SeverityCell,
|
|
},
|
|
{ Header: 'Status', accessor: 'value', show: options.statusField, width: 100, Cell: statusCell },
|
|
{ Header: 'Problem', accessor: 'description', minWidth: 200, Cell: ProblemCell},
|
|
{ Header: 'Tags', accessor: 'tags', show: options.showTags, className: 'problem-tags', Cell: TagCell },
|
|
{ Header: 'Time', className: 'last-change', width: timeColWidth,
|
|
accessor: 'lastchangeUnix',
|
|
id: 'lastchange',
|
|
Cell: row => row.original.lastchange,
|
|
},
|
|
{ Header: 'Details', className: 'custom-expander', width: 60, expander: true, Expander: CustomExpander },
|
|
];
|
|
for (const column of columns) {
|
|
if (column.show || column.show === undefined) {
|
|
delete column.show;
|
|
result.push(column);
|
|
}
|
|
}
|
|
return result;
|
|
}
|
|
|
|
getExpandedPage = (page: number) => {
|
|
return this.state.expanded[page] || {};
|
|
}
|
|
|
|
handleExpandedChange = expanded => {
|
|
const nextExpanded = {...this.state.expanded};
|
|
nextExpanded[this.state.page] = expanded;
|
|
this.setState({
|
|
expanded: nextExpanded
|
|
});
|
|
}
|
|
|
|
render() {
|
|
// console.log(this.props.problems);
|
|
const columns = this.buildColumns();
|
|
this.rootWidth = this.rootRef && this.rootRef.clientWidth;
|
|
|
|
return (
|
|
<div className="panel-problems" ref={this.setRootRef}>
|
|
<ReactTable
|
|
data={this.props.problems}
|
|
columns={columns}
|
|
defaultPageSize={10}
|
|
loading={this.props.loading}
|
|
SubComponent={props =>
|
|
<ProblemDetails {...props}
|
|
rootWidth={this.rootWidth}
|
|
timeRange={this.props.timeRange}
|
|
getProblemEvents={this.props.getProblemEvents}
|
|
onProblemAck={this.handleProblemAck}
|
|
/>
|
|
}
|
|
expanded={this.getExpandedPage(this.state.page)}
|
|
onExpandedChange={this.handleExpandedChange}
|
|
onPageChange={page => this.setState({ page })}
|
|
/>
|
|
</div>
|
|
);
|
|
}
|
|
}
|
|
|
|
function SeverityCell(props: RTCell<Trigger>) {
|
|
return (
|
|
<div className='severity-cell' style={{ background: props.original.color }}>
|
|
{props.original.severity}
|
|
</div>
|
|
);
|
|
}
|
|
|
|
const DEFAULT_OK_COLOR = 'rgb(56, 189, 113)';
|
|
const DEFAULT_PROBLEM_COLOR = 'rgb(215, 0, 0)';
|
|
|
|
function StatusCell(props: RTCell<Trigger>, okColor = DEFAULT_OK_COLOR, problemColor = DEFAULT_PROBLEM_COLOR, highlightNewerThan?: string) {
|
|
// console.log(props);
|
|
const status = props.value === '0' ? 'RESOLVED' : 'PROBLEM';
|
|
const color = props.value === '0' ? okColor : problemColor;
|
|
let newProblem = false;
|
|
if (highlightNewerThan) {
|
|
newProblem = isNewProblem(props.original, highlightNewerThan);
|
|
}
|
|
return (
|
|
<span className={newProblem ? 'problem-status--new' : ''} style={{ color }}>{status}</span>
|
|
);
|
|
}
|
|
|
|
function GroupCell(props: RTCell<Trigger>) {
|
|
let groups = "";
|
|
if (props.value && props.value.length) {
|
|
groups = props.value.map(g => g.name).join(', ');
|
|
}
|
|
return (
|
|
<span>{groups}</span>
|
|
);
|
|
}
|
|
|
|
function ProblemCell(props: RTCell<Trigger>) {
|
|
const comments = props.original.comments;
|
|
return (
|
|
<div>
|
|
<span className="problem-description">{props.value}</span>
|
|
{/* {comments && <FAIcon icon="file-text-o" customClass="comments-icon" />} */}
|
|
</div>
|
|
);
|
|
}
|
|
|
|
function TagCell(props: RTCell<Trigger>) {
|
|
const tags = props.value || [];
|
|
return [
|
|
tags.map(tag => <EventTag key={tag.tag + tag.value} tag={tag} />)
|
|
];
|
|
}
|
|
|
|
function CustomExpander(props: RTCell<any>) {
|
|
return (
|
|
<span className={props.isExpanded ? "expanded" : ""}>
|
|
<i className="fa fa-info-circle"></i>
|
|
</span>
|
|
);
|
|
}
|
|
|
|
function isNewProblem(problem: Trigger, highlightNewerThan: string): boolean {
|
|
try {
|
|
const highlightIntervalMs = utils.parseInterval(highlightNewerThan);
|
|
const durationSec = (Date.now() - problem.lastchangeUnix * 1000);
|
|
return durationSec < highlightIntervalMs;
|
|
} catch (e) {
|
|
return false;
|
|
}
|
|
}
|