grafana/public/app/plugins/datasource/stackdriver/components/Metrics.tsx

200 lines
5.9 KiB
TypeScript
Raw Normal View History

2018-12-13 07:16:48 +08:00
import React from 'react';
import _ from 'lodash';
2019-01-11 20:53:04 +08:00
import StackdriverDatasource from '../datasource';
import appEvents from 'app/core/app_events';
2019-01-08 21:03:52 +08:00
import { MetricDescriptor } from '../types';
2019-01-02 22:07:38 +08:00
import { MetricSelect } from 'app/core/components/Select/MetricSelect';
import { TemplateSrv } from 'app/features/templating/template_srv';
2018-12-13 07:16:48 +08:00
export interface Props {
2019-01-08 21:03:52 +08:00
onChange: (metricDescriptor: MetricDescriptor) => void;
templateSrv: TemplateSrv;
2019-01-11 20:53:04 +08:00
datasource: StackdriverDatasource;
2018-12-13 07:16:48 +08:00
defaultProject: string;
metricType: string;
2018-12-19 22:54:45 +08:00
children?: (renderProps: any) => JSX.Element;
2018-12-13 07:16:48 +08:00
}
interface State {
2019-01-08 21:03:52 +08:00
metricDescriptors: MetricDescriptor[];
2018-12-13 07:16:48 +08:00
metrics: any[];
services: any[];
service: string;
metric: string;
2019-01-08 21:03:52 +08:00
metricDescriptor: MetricDescriptor;
2018-12-29 02:45:24 +08:00
defaultProject: string;
2018-12-13 07:16:48 +08:00
}
2018-12-19 21:29:00 +08:00
export class Metrics extends React.Component<Props, State> {
2018-12-13 07:16:48 +08:00
state: State = {
metricDescriptors: [],
metrics: [],
services: [],
service: '',
metric: '',
2018-12-19 22:54:45 +08:00
metricDescriptor: null,
2018-12-29 02:45:24 +08:00
defaultProject: '',
2018-12-13 07:16:48 +08:00
};
constructor(props) {
super(props);
}
componentDidMount() {
2018-12-29 02:45:24 +08:00
this.setState({ defaultProject: this.props.defaultProject }, () => {
this.getCurrentProject()
.then(this.loadMetricDescriptors.bind(this))
.then(this.initializeServiceAndMetrics.bind(this));
});
2018-12-13 07:16:48 +08:00
}
async getCurrentProject() {
return new Promise(async (resolve, reject) => {
try {
2018-12-29 02:45:24 +08:00
if (!this.state.defaultProject || this.state.defaultProject === 'loading project...') {
const defaultProject = await this.props.datasource.getDefaultProject();
this.setState({ defaultProject });
2018-12-13 07:16:48 +08:00
}
2018-12-29 02:45:24 +08:00
resolve(this.state.defaultProject);
2018-12-13 07:16:48 +08:00
} catch (error) {
appEvents.emit('ds-request-error', error);
2018-12-13 07:16:48 +08:00
reject();
}
});
}
async loadMetricDescriptors() {
2018-12-29 02:45:24 +08:00
if (this.state.defaultProject !== 'loading project...') {
const metricDescriptors = await this.props.datasource.getMetricTypes(this.state.defaultProject);
2018-12-13 07:16:48 +08:00
this.setState({ metricDescriptors });
return metricDescriptors;
} else {
return [];
}
}
async initializeServiceAndMetrics() {
const { metricDescriptors } = this.state;
const services = this.getServicesList(metricDescriptors);
const metrics = this.getMetricsList(metricDescriptors);
const service = metrics.length > 0 ? metrics[0].service : '';
2018-12-19 22:54:45 +08:00
const metricDescriptor = this.getSelectedMetricDescriptor(this.props.metricType);
this.setState({ metricDescriptors, services, metrics, service: service, metricDescriptor });
}
getSelectedMetricDescriptor(metricType) {
return this.state.metricDescriptors.find(md => md.type === this.props.templateSrv.replace(metricType));
2018-12-13 07:16:48 +08:00
}
2019-01-08 21:03:52 +08:00
getMetricsList(metricDescriptors: MetricDescriptor[]) {
2018-12-19 22:54:45 +08:00
const selectedMetricDescriptor = this.getSelectedMetricDescriptor(this.props.metricType);
2018-12-29 02:45:24 +08:00
if (!selectedMetricDescriptor) {
return [];
}
const metricsByService = metricDescriptors
.filter(m => m.service === selectedMetricDescriptor.service)
.map(m => ({
service: m.service,
value: m.type,
label: m.displayName,
description: m.description,
}));
2018-12-13 07:16:48 +08:00
return metricsByService;
}
onServiceChange = service => {
2018-12-13 07:16:48 +08:00
const { metricDescriptors } = this.state;
const { templateSrv, metricType } = this.props;
const metrics = metricDescriptors
.filter(m => m.service === templateSrv.replace(service))
.map(m => ({
service: m.service,
value: m.type,
label: m.displayName,
description: m.description,
}));
2018-12-13 07:16:48 +08:00
this.setState({ service, metrics });
if (metrics.length > 0 && !metrics.some(m => m.value === templateSrv.replace(metricType))) {
this.onMetricTypeChange(metrics[0].value);
2018-12-13 07:16:48 +08:00
}
};
2018-12-13 07:16:48 +08:00
onMetricTypeChange = value => {
2018-12-19 22:54:45 +08:00
const metricDescriptor = this.getSelectedMetricDescriptor(value);
this.setState({ metricDescriptor });
2019-01-03 16:15:49 +08:00
this.props.onChange({ ...metricDescriptor, type: value });
};
2018-12-13 07:16:48 +08:00
2019-01-08 21:03:52 +08:00
getServicesList(metricDescriptors: MetricDescriptor[]) {
2018-12-13 07:16:48 +08:00
const services = metricDescriptors.map(m => ({
value: m.service,
label: _.startCase(m.serviceShortName),
}));
return services.length > 0 ? _.uniqBy(services, s => s.value) : [];
}
getTemplateVariablesGroup() {
return {
label: 'Template Variables',
options: this.props.templateSrv.variables.map(v => ({
label: `$${v.name}`,
value: `$${v.name}`,
})),
};
}
render() {
2018-12-19 21:29:00 +08:00
const { services, service, metrics } = this.state;
const { metricType, templateSrv } = this.props;
2018-12-13 07:16:48 +08:00
return (
2019-01-08 20:00:31 +08:00
<>
2018-12-13 07:16:48 +08:00
<div className="gf-form-inline">
<div className="gf-form">
<span className="gf-form-label width-9 query-keyword">Service</span>
2019-01-02 22:07:38 +08:00
<MetricSelect
onChange={this.onServiceChange}
2019-01-02 22:07:38 +08:00
value={service}
2018-12-13 07:16:48 +08:00
options={services}
2019-01-02 22:07:38 +08:00
isSearchable={false}
2018-12-13 07:16:48 +08:00
placeholder="Select Services"
className="width-15"
/>
</div>
<div className="gf-form gf-form--grow">
<div className="gf-form-label gf-form-label--grow" />
</div>
</div>
<div className="gf-form-inline">
<div className="gf-form">
<span className="gf-form-label width-9 query-keyword">Metric</span>
2019-01-02 22:07:38 +08:00
<MetricSelect
onChange={this.onMetricTypeChange}
2019-01-02 22:07:38 +08:00
value={metricType}
variables={templateSrv.variables}
2019-01-03 16:15:49 +08:00
options={[
{
label: 'Metrics',
expanded: true,
options: metrics,
},
]}
2018-12-13 07:16:48 +08:00
placeholder="Select Metric"
className="width-15"
/>
</div>
<div className="gf-form gf-form--grow">
<div className="gf-form-label gf-form-label--grow" />
</div>
</div>
2018-12-19 22:54:45 +08:00
{this.props.children(this.state.metricDescriptor)}
2019-01-08 20:00:31 +08:00
</>
2018-12-13 07:16:48 +08:00
);
}
}