The open and composable observability and data visualization platform. Visualize metrics, logs, and traces from multiple sources like Prometheus, Loki, Elasticsearch, InfluxDB, Postgres and many more.
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.
 
 
 
 
 
 
grafana/public/app/features/alerting/TestRuleResult.tsx

129 lines
3.1 KiB

import React, { PureComponent } from 'react';
import { getBackendSrv } from '@grafana/runtime';
import {
LoadingPlaceholder,
JSONFormatter,
Icon,
HorizontalGroup,
ClipboardButton,
clearButtonStyles,
withTheme2,
Themeable2,
} from '@grafana/ui';
import { DashboardModel, PanelModel } from '../dashboard/state';
export interface Props extends Themeable2 {
dashboard: DashboardModel;
panel: PanelModel;
}
interface State {
isLoading: boolean;
allNodesExpanded: boolean | null;
testRuleResponse: {};
}
class UnThemedTestRuleResult extends PureComponent<Props, State> {
readonly state: State = {
isLoading: false,
allNodesExpanded: null,
testRuleResponse: {},
};
formattedJson: any;
clipboard: any;
componentDidMount() {
this.testRule();
}
async testRule() {
const { dashboard, panel } = this.props;
// dashboard save model
const model = dashboard.getSaveModelClone();
// now replace panel to get current edits
model.panels = model.panels.map((dashPanel) => {
return dashPanel.id === panel.id ? panel.getSaveModel() : dashPanel;
});
const payload = { dashboard: model, panelId: panel.id };
this.setState({ isLoading: true });
const testRuleResponse = await getBackendSrv().post(`/api/alerts/test`, payload);
this.setState({ isLoading: false, testRuleResponse });
}
setFormattedJson = (formattedJson: any) => {
this.formattedJson = formattedJson;
};
getTextForClipboard = () => {
return JSON.stringify(this.formattedJson, null, 2);
};
onToggleExpand = () => {
this.setState((prevState) => ({
...prevState,
allNodesExpanded: !this.state.allNodesExpanded,
}));
};
getNrOfOpenNodes = () => {
if (this.state.allNodesExpanded === null) {
return 3; // 3 is default, ie when state is null
} else if (this.state.allNodesExpanded) {
return 20;
}
return 1;
};
renderExpandCollapse = () => {
const { allNodesExpanded } = this.state;
const collapse = (
<>
<Icon name="minus-circle" /> Collapse All
</>
);
const expand = (
<>
<Icon name="plus-circle" /> Expand All
</>
);
return allNodesExpanded ? collapse : expand;
};
render() {
const { testRuleResponse, isLoading } = this.state;
const clearButton = clearButtonStyles(this.props.theme);
if (isLoading === true) {
return <LoadingPlaceholder text="Evaluating rule" />;
}
const openNodes = this.getNrOfOpenNodes();
return (
<>
<div className="pull-right">
<HorizontalGroup spacing="md">
<button type="button" className={clearButton} onClick={this.onToggleExpand}>
{this.renderExpandCollapse()}
</button>
<ClipboardButton getText={this.getTextForClipboard} icon="copy">
Copy to Clipboard
</ClipboardButton>
</HorizontalGroup>
</div>
<JSONFormatter json={testRuleResponse} open={openNodes} onDidRender={this.setFormattedJson} />
</>
);
}
}
export const TestRuleResult = withTheme2(UnThemedTestRuleResult);