2019-10-17 05:38:09 -07:00
|
|
|
import React, { Component } from 'react';
|
|
|
|
|
2019-10-28 07:02:42 -07:00
|
|
|
import { Alert, Button, Col, Nav, NavItem, NavLink, Row, TabContent, TabPane } from 'reactstrap';
|
2019-10-17 05:38:09 -07:00
|
|
|
|
|
|
|
import moment from 'moment-timezone';
|
|
|
|
|
|
|
|
import ExpressionInput from './ExpressionInput';
|
2020-01-14 10:34:48 -08:00
|
|
|
import GraphControls from './GraphControls';
|
|
|
|
import { GraphTabContent } from './GraphTabContent';
|
2019-10-17 05:38:09 -07:00
|
|
|
import DataTable from './DataTable';
|
|
|
|
import TimeInput from './TimeInput';
|
2019-10-23 03:47:37 -07:00
|
|
|
import QueryStatsView, { QueryStats } from './QueryStatsView';
|
2020-01-14 10:34:48 -08:00
|
|
|
import { QueryParams } from '../../types/types';
|
2020-10-22 08:22:32 -07:00
|
|
|
import { API_PATH } from '../../constants/constants';
|
2019-10-17 05:38:09 -07:00
|
|
|
|
|
|
|
interface PanelProps {
|
|
|
|
options: PanelOptions;
|
|
|
|
onOptionsChanged: (opts: PanelOptions) => void;
|
2020-01-24 14:44:18 -08:00
|
|
|
useLocalTime: boolean;
|
2019-10-26 10:50:22 -07:00
|
|
|
pastQueries: string[];
|
2019-10-17 05:38:09 -07:00
|
|
|
metricNames: string[];
|
|
|
|
removePanel: () => void;
|
2019-10-23 13:18:41 -07:00
|
|
|
onExecuteQuery: (query: string) => void;
|
2020-10-22 08:22:32 -07:00
|
|
|
pathPrefix: string;
|
2020-11-02 07:16:54 -08:00
|
|
|
enableMetricAutocomplete: boolean;
|
2019-10-17 05:38:09 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
interface PanelState {
|
|
|
|
data: any; // TODO: Type data.
|
2019-11-24 04:14:57 -08:00
|
|
|
lastQueryParams: QueryParams | null;
|
2019-10-17 05:38:09 -07:00
|
|
|
loading: boolean;
|
|
|
|
error: string | null;
|
2019-10-28 07:02:42 -07:00
|
|
|
stats: QueryStats | null;
|
2020-01-18 14:40:25 -08:00
|
|
|
exprInputValue: string;
|
2019-10-17 05:38:09 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
export interface PanelOptions {
|
|
|
|
expr: string;
|
|
|
|
type: PanelType;
|
2020-08-21 02:53:11 -07:00
|
|
|
range: number; // Range in milliseconds.
|
2019-10-17 05:38:09 -07:00
|
|
|
endTime: number | null; // Timestamp in milliseconds.
|
|
|
|
resolution: number | null; // Resolution in seconds.
|
|
|
|
stacked: boolean;
|
|
|
|
}
|
|
|
|
|
|
|
|
export enum PanelType {
|
|
|
|
Graph = 'graph',
|
|
|
|
Table = 'table',
|
|
|
|
}
|
|
|
|
|
|
|
|
export const PanelDefaultOptions: PanelOptions = {
|
|
|
|
type: PanelType.Table,
|
|
|
|
expr: '',
|
2020-08-21 02:53:11 -07:00
|
|
|
range: 60 * 60 * 1000,
|
2019-10-17 05:38:09 -07:00
|
|
|
endTime: null,
|
|
|
|
resolution: null,
|
|
|
|
stacked: false,
|
2019-10-28 07:02:42 -07:00
|
|
|
};
|
2019-10-17 05:38:09 -07:00
|
|
|
|
2020-10-22 08:22:32 -07:00
|
|
|
class Panel extends Component<PanelProps, PanelState> {
|
2019-10-17 05:38:09 -07:00
|
|
|
private abortInFlightFetch: (() => void) | null = null;
|
|
|
|
|
|
|
|
constructor(props: PanelProps) {
|
|
|
|
super(props);
|
|
|
|
|
|
|
|
this.state = {
|
|
|
|
data: null,
|
|
|
|
lastQueryParams: null,
|
|
|
|
loading: false,
|
|
|
|
error: null,
|
|
|
|
stats: null,
|
2020-01-18 14:40:25 -08:00
|
|
|
exprInputValue: props.options.expr,
|
2019-10-17 05:38:09 -07:00
|
|
|
};
|
|
|
|
}
|
|
|
|
|
2019-12-12 14:22:12 -08:00
|
|
|
componentDidUpdate({ options: prevOpts }: PanelProps) {
|
|
|
|
const { endTime, range, resolution, type } = this.props.options;
|
2019-10-28 07:02:42 -07:00
|
|
|
if (
|
2019-12-12 14:22:12 -08:00
|
|
|
prevOpts.endTime !== endTime ||
|
|
|
|
prevOpts.range !== range ||
|
|
|
|
prevOpts.resolution !== resolution ||
|
|
|
|
prevOpts.type !== type
|
2019-10-28 07:02:42 -07:00
|
|
|
) {
|
2019-12-12 14:22:12 -08:00
|
|
|
this.executeQuery();
|
2019-10-17 05:38:09 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
componentDidMount() {
|
2019-12-12 14:22:12 -08:00
|
|
|
this.executeQuery();
|
2019-10-17 05:38:09 -07:00
|
|
|
}
|
|
|
|
|
2019-12-12 14:22:12 -08:00
|
|
|
executeQuery = (): void => {
|
2020-01-18 14:40:25 -08:00
|
|
|
const { exprInputValue: expr } = this.state;
|
2019-10-23 03:47:37 -07:00
|
|
|
const queryStart = Date.now();
|
2019-10-28 07:02:42 -07:00
|
|
|
this.props.onExecuteQuery(expr);
|
2020-01-18 14:40:25 -08:00
|
|
|
if (this.props.options.expr !== expr) {
|
|
|
|
this.setOptions({ expr });
|
|
|
|
}
|
2019-10-17 05:38:09 -07:00
|
|
|
if (expr === '') {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (this.abortInFlightFetch) {
|
|
|
|
this.abortInFlightFetch();
|
|
|
|
this.abortInFlightFetch = null;
|
|
|
|
}
|
|
|
|
|
|
|
|
const abortController = new AbortController();
|
|
|
|
this.abortInFlightFetch = () => abortController.abort();
|
2019-10-28 07:02:42 -07:00
|
|
|
this.setState({ loading: true });
|
2019-10-17 05:38:09 -07:00
|
|
|
|
|
|
|
const endTime = this.getEndTime().valueOf() / 1000; // TODO: shouldn't valueof only work when it's a moment?
|
2020-08-21 02:53:11 -07:00
|
|
|
const startTime = endTime - this.props.options.range / 1000;
|
|
|
|
const resolution = this.props.options.resolution || Math.max(Math.floor(this.props.options.range / 250000), 1);
|
2019-11-03 04:21:35 -08:00
|
|
|
const params: URLSearchParams = new URLSearchParams({
|
2019-10-28 07:02:42 -07:00
|
|
|
query: expr,
|
2019-11-03 04:21:35 -08:00
|
|
|
});
|
2019-10-17 05:38:09 -07:00
|
|
|
|
2019-11-03 04:21:35 -08:00
|
|
|
let path: string;
|
2019-10-17 05:38:09 -07:00
|
|
|
switch (this.props.options.type) {
|
|
|
|
case 'graph':
|
2020-10-22 08:22:32 -07:00
|
|
|
path = 'query_range';
|
2019-11-03 04:21:35 -08:00
|
|
|
params.append('start', startTime.toString());
|
|
|
|
params.append('end', endTime.toString());
|
|
|
|
params.append('step', resolution.toString());
|
2019-10-17 05:38:09 -07:00
|
|
|
break;
|
|
|
|
case 'table':
|
2020-10-22 08:22:32 -07:00
|
|
|
path = 'query';
|
2019-11-03 04:21:35 -08:00
|
|
|
params.append('time', endTime.toString());
|
2019-10-17 05:38:09 -07:00
|
|
|
break;
|
|
|
|
default:
|
|
|
|
throw new Error('Invalid panel type "' + this.props.options.type + '"');
|
|
|
|
}
|
|
|
|
|
2020-10-22 08:22:32 -07:00
|
|
|
fetch(`${this.props.pathPrefix}/${API_PATH}/${path}?${params}`, {
|
2020-01-20 07:50:32 -08:00
|
|
|
cache: 'no-store',
|
|
|
|
credentials: 'same-origin',
|
|
|
|
signal: abortController.signal,
|
|
|
|
})
|
2019-10-28 07:02:42 -07:00
|
|
|
.then(resp => resp.json())
|
|
|
|
.then(json => {
|
|
|
|
if (json.status !== 'success') {
|
|
|
|
throw new Error(json.error || 'invalid response JSON');
|
|
|
|
}
|
2019-10-17 05:38:09 -07:00
|
|
|
|
2019-10-28 07:02:42 -07:00
|
|
|
let resultSeries = 0;
|
|
|
|
if (json.data) {
|
|
|
|
const { resultType, result } = json.data;
|
|
|
|
if (resultType === 'scalar') {
|
|
|
|
resultSeries = 1;
|
|
|
|
} else if (result && result.length > 0) {
|
|
|
|
resultSeries = result.length;
|
|
|
|
}
|
2019-10-23 03:47:37 -07:00
|
|
|
}
|
|
|
|
|
2019-10-28 07:02:42 -07:00
|
|
|
this.setState({
|
|
|
|
error: null,
|
|
|
|
data: json.data,
|
|
|
|
lastQueryParams: {
|
|
|
|
startTime,
|
|
|
|
endTime,
|
|
|
|
resolution,
|
|
|
|
},
|
|
|
|
stats: {
|
|
|
|
loadTime: Date.now() - queryStart,
|
|
|
|
resolution,
|
|
|
|
resultSeries,
|
|
|
|
},
|
|
|
|
loading: false,
|
|
|
|
});
|
|
|
|
this.abortInFlightFetch = null;
|
2019-10-17 05:38:09 -07:00
|
|
|
})
|
2019-10-28 07:02:42 -07:00
|
|
|
.catch(error => {
|
|
|
|
if (error.name === 'AbortError') {
|
|
|
|
// Aborts are expected, don't show an error for them.
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
this.setState({
|
|
|
|
error: 'Error executing query: ' + error.message,
|
|
|
|
loading: false,
|
|
|
|
});
|
|
|
|
});
|
|
|
|
};
|
2019-10-17 05:38:09 -07:00
|
|
|
|
|
|
|
setOptions(opts: object): void {
|
2019-10-28 07:02:42 -07:00
|
|
|
const newOpts = { ...this.props.options, ...opts };
|
2019-10-17 05:38:09 -07:00
|
|
|
this.props.onOptionsChanged(newOpts);
|
|
|
|
}
|
|
|
|
|
|
|
|
handleExpressionChange = (expr: string): void => {
|
2020-01-18 14:40:25 -08:00
|
|
|
this.setState({ exprInputValue: expr });
|
2019-10-28 07:02:42 -07:00
|
|
|
};
|
2019-10-17 05:38:09 -07:00
|
|
|
|
|
|
|
handleChangeRange = (range: number): void => {
|
2019-10-28 07:02:42 -07:00
|
|
|
this.setOptions({ range: range });
|
|
|
|
};
|
2019-10-17 05:38:09 -07:00
|
|
|
|
|
|
|
getEndTime = (): number | moment.Moment => {
|
|
|
|
if (this.props.options.endTime === null) {
|
|
|
|
return moment();
|
|
|
|
}
|
|
|
|
return this.props.options.endTime;
|
2019-10-28 07:02:42 -07:00
|
|
|
};
|
2019-10-17 05:38:09 -07:00
|
|
|
|
|
|
|
handleChangeEndTime = (endTime: number | null) => {
|
2019-10-28 07:02:42 -07:00
|
|
|
this.setOptions({ endTime: endTime });
|
|
|
|
};
|
2019-10-17 05:38:09 -07:00
|
|
|
|
|
|
|
handleChangeResolution = (resolution: number | null) => {
|
2019-10-28 07:02:42 -07:00
|
|
|
this.setOptions({ resolution: resolution });
|
|
|
|
};
|
2019-10-17 05:38:09 -07:00
|
|
|
|
2019-12-12 14:22:12 -08:00
|
|
|
handleChangeType = (type: PanelType) => {
|
2020-05-15 05:03:15 -07:00
|
|
|
if (this.props.options.type === type) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2019-12-12 14:22:12 -08:00
|
|
|
this.setState({ data: null });
|
|
|
|
this.setOptions({ type: type });
|
|
|
|
};
|
|
|
|
|
2019-10-17 05:38:09 -07:00
|
|
|
handleChangeStacking = (stacked: boolean) => {
|
2019-10-28 07:02:42 -07:00
|
|
|
this.setOptions({ stacked: stacked });
|
|
|
|
};
|
2019-10-17 05:38:09 -07:00
|
|
|
|
|
|
|
render() {
|
2019-10-26 10:50:22 -07:00
|
|
|
const { pastQueries, metricNames, options } = this.props;
|
2019-10-17 05:38:09 -07:00
|
|
|
return (
|
|
|
|
<div className="panel">
|
|
|
|
<Row>
|
|
|
|
<Col>
|
|
|
|
<ExpressionInput
|
2020-01-18 14:40:25 -08:00
|
|
|
value={this.state.exprInputValue}
|
2019-12-12 14:22:12 -08:00
|
|
|
onExpressionChange={this.handleExpressionChange}
|
2019-10-17 05:38:09 -07:00
|
|
|
executeQuery={this.executeQuery}
|
|
|
|
loading={this.state.loading}
|
2020-11-02 07:16:54 -08:00
|
|
|
enableMetricAutocomplete={this.props.enableMetricAutocomplete}
|
2019-10-26 10:50:22 -07:00
|
|
|
autocompleteSections={{
|
|
|
|
'Query History': pastQueries,
|
|
|
|
'Metric Names': metricNames,
|
|
|
|
}}
|
2019-10-17 05:38:09 -07:00
|
|
|
/>
|
|
|
|
</Col>
|
|
|
|
</Row>
|
|
|
|
<Row>
|
2019-10-28 07:02:42 -07:00
|
|
|
<Col>{this.state.error && <Alert color="danger">{this.state.error}</Alert>}</Col>
|
2019-10-17 05:38:09 -07:00
|
|
|
</Row>
|
|
|
|
<Row>
|
|
|
|
<Col>
|
|
|
|
<Nav tabs>
|
|
|
|
<NavItem>
|
|
|
|
<NavLink
|
2019-10-26 10:50:22 -07:00
|
|
|
className={options.type === 'table' ? 'active' : ''}
|
2019-12-12 14:22:12 -08:00
|
|
|
onClick={() => this.handleChangeType(PanelType.Table)}
|
2019-10-17 05:38:09 -07:00
|
|
|
>
|
|
|
|
Table
|
|
|
|
</NavLink>
|
|
|
|
</NavItem>
|
|
|
|
<NavItem>
|
|
|
|
<NavLink
|
2019-10-26 10:50:22 -07:00
|
|
|
className={options.type === 'graph' ? 'active' : ''}
|
2019-12-12 14:22:12 -08:00
|
|
|
onClick={() => this.handleChangeType(PanelType.Graph)}
|
2019-10-17 05:38:09 -07:00
|
|
|
>
|
|
|
|
Graph
|
|
|
|
</NavLink>
|
|
|
|
</NavItem>
|
2019-10-28 07:02:42 -07:00
|
|
|
{!this.state.loading && !this.state.error && this.state.stats && <QueryStatsView {...this.state.stats} />}
|
2019-10-17 05:38:09 -07:00
|
|
|
</Nav>
|
2019-10-26 10:50:22 -07:00
|
|
|
<TabContent activeTab={options.type}>
|
2019-10-17 05:38:09 -07:00
|
|
|
<TabPane tabId="table">
|
2019-10-28 07:02:42 -07:00
|
|
|
{options.type === 'table' && (
|
2019-10-17 05:38:09 -07:00
|
|
|
<>
|
|
|
|
<div className="table-controls">
|
|
|
|
<TimeInput
|
2019-10-26 10:50:22 -07:00
|
|
|
time={options.endTime}
|
2020-01-24 14:44:18 -08:00
|
|
|
useLocalTime={this.props.useLocalTime}
|
2019-10-26 10:50:22 -07:00
|
|
|
range={options.range}
|
2019-10-17 05:38:09 -07:00
|
|
|
placeholder="Evaluation time"
|
|
|
|
onChangeTime={this.handleChangeEndTime}
|
|
|
|
/>
|
|
|
|
</div>
|
|
|
|
<DataTable data={this.state.data} />
|
|
|
|
</>
|
2019-10-28 07:02:42 -07:00
|
|
|
)}
|
2019-10-17 05:38:09 -07:00
|
|
|
</TabPane>
|
|
|
|
<TabPane tabId="graph">
|
2019-10-28 07:02:42 -07:00
|
|
|
{this.props.options.type === 'graph' && (
|
2019-10-17 05:38:09 -07:00
|
|
|
<>
|
|
|
|
<GraphControls
|
2019-10-26 10:50:22 -07:00
|
|
|
range={options.range}
|
|
|
|
endTime={options.endTime}
|
2020-01-24 14:44:18 -08:00
|
|
|
useLocalTime={this.props.useLocalTime}
|
2019-10-26 10:50:22 -07:00
|
|
|
resolution={options.resolution}
|
|
|
|
stacked={options.stacked}
|
2019-10-17 05:38:09 -07:00
|
|
|
onChangeRange={this.handleChangeRange}
|
|
|
|
onChangeEndTime={this.handleChangeEndTime}
|
|
|
|
onChangeResolution={this.handleChangeResolution}
|
|
|
|
onChangeStacking={this.handleChangeStacking}
|
|
|
|
/>
|
2019-11-26 11:48:53 -08:00
|
|
|
<GraphTabContent
|
|
|
|
data={this.state.data}
|
|
|
|
stacked={options.stacked}
|
2020-01-24 14:44:18 -08:00
|
|
|
useLocalTime={this.props.useLocalTime}
|
2019-11-26 11:48:53 -08:00
|
|
|
lastQueryParams={this.state.lastQueryParams}
|
|
|
|
/>
|
2019-10-17 05:38:09 -07:00
|
|
|
</>
|
2019-10-28 07:02:42 -07:00
|
|
|
)}
|
2019-10-17 05:38:09 -07:00
|
|
|
</TabPane>
|
|
|
|
</TabContent>
|
|
|
|
</Col>
|
|
|
|
</Row>
|
|
|
|
<Row>
|
|
|
|
<Col>
|
2019-10-28 07:02:42 -07:00
|
|
|
<Button className="float-right" color="link" onClick={this.props.removePanel} size="sm">
|
|
|
|
Remove Panel
|
|
|
|
</Button>
|
2019-10-17 05:38:09 -07:00
|
|
|
</Col>
|
|
|
|
</Row>
|
|
|
|
</div>
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
export default Panel;
|