EVOLUTION-MANAGER
Edit File: datasource.ts
import { stackdriverUnitMappings } from './constants'; import appEvents from 'app/core/app_events'; import _ from 'lodash'; import StackdriverMetricFindQuery from './StackdriverMetricFindQuery'; import { StackdriverQuery, MetricDescriptor, StackdriverOptions, Filter } from './types'; import { DataSourceApi, DataQueryRequest, DataSourceInstanceSettings, ScopedVars } from '@grafana/data'; import { BackendSrv } from 'app/core/services/backend_srv'; import { TemplateSrv } from 'app/features/templating/template_srv'; import { TimeSrv } from 'app/features/dashboard/services/TimeSrv'; import { CoreEvents } from 'app/types'; export default class StackdriverDatasource extends DataSourceApi<StackdriverQuery, StackdriverOptions> { url: string; baseUrl: string; projectName: string; authenticationType: string; queryPromise: Promise<any>; metricTypes: any[]; /** @ngInject */ constructor( instanceSettings: DataSourceInstanceSettings<StackdriverOptions>, private backendSrv: BackendSrv, public templateSrv: TemplateSrv, private timeSrv: TimeSrv ) { super(instanceSettings); this.baseUrl = `/stackdriver/`; this.url = instanceSettings.url; this.projectName = instanceSettings.jsonData.defaultProject || ''; this.authenticationType = instanceSettings.jsonData.authenticationType || 'jwt'; this.metricTypes = []; } get variables() { return this.templateSrv.variables.map(v => `$${v.name}`); } async getTimeSeries(options: any) { const queries = options.targets .filter((target: any) => { return !target.hide && target.metricType; }) .map((t: any) => { return { refId: t.refId, intervalMs: options.intervalMs, datasourceId: this.id, metricType: this.templateSrv.replace(t.metricType, options.scopedVars || {}), crossSeriesReducer: this.templateSrv.replace(t.crossSeriesReducer || 'REDUCE_MEAN', options.scopedVars || {}), perSeriesAligner: this.templateSrv.replace(t.perSeriesAligner, options.scopedVars || {}), alignmentPeriod: this.templateSrv.replace(t.alignmentPeriod, options.scopedVars || {}), groupBys: this.interpolateGroupBys(t.groupBys, options.scopedVars), view: t.view || 'FULL', filters: this.interpolateFilters(t.filters, options.scopedVars), aliasBy: this.templateSrv.replace(t.aliasBy, options.scopedVars || {}), type: 'timeSeriesQuery', }; }); if (queries.length > 0) { const { data } = await this.backendSrv.datasourceRequest({ url: '/api/tsdb/query', method: 'POST', data: { from: options.range.from.valueOf().toString(), to: options.range.to.valueOf().toString(), queries, }, }); return data; } else { return { results: [] }; } } interpolateFilters(filters: string[], scopedVars: ScopedVars) { const completeFilter = _.chunk(filters, 4) .map(([key, operator, value, condition = 'AND']) => ({ key, operator, value, condition, })) .reduce((res, filter) => (filter.value ? [...res, filter] : res), []); const filterArray = _.flatten( completeFilter.map(({ key, operator, value, condition }: Filter) => [key, operator, value, condition]) ); return (filterArray || []).map(f => { return this.templateSrv.replace(f, scopedVars || {}, 'regex'); }); } async getLabels(metricType: string, refId: string, groupBys?: string[]) { const response = await this.getTimeSeries({ targets: [ { refId: refId, datasourceId: this.id, metricType: this.templateSrv.replace(metricType), groupBys: this.interpolateGroupBys(groupBys || [], {}), crossSeriesReducer: 'REDUCE_NONE', view: 'HEADERS', }, ], range: this.timeSrv.timeRange(), }); const result = response.results[refId]; return result && result.meta ? result.meta.labels : {}; } interpolateGroupBys(groupBys: string[], scopedVars: {}): string[] { let interpolatedGroupBys: any[] = []; (groupBys || []).forEach(gb => { const interpolated = this.templateSrv.replace(gb, scopedVars || {}, 'csv').split(','); if (Array.isArray(interpolated)) { interpolatedGroupBys = interpolatedGroupBys.concat(interpolated); } else { interpolatedGroupBys.push(interpolated); } }); return interpolatedGroupBys; } resolvePanelUnitFromTargets(targets: any[]) { let unit; if (targets.length > 0 && targets.every(t => t.unit === targets[0].unit)) { if (stackdriverUnitMappings.hasOwnProperty(targets[0].unit)) { // @ts-ignore unit = stackdriverUnitMappings[targets[0].unit]; } } return unit; } async query(options: DataQueryRequest<StackdriverQuery>) { const result: any[] = []; const data = await this.getTimeSeries(options); if (data.results) { Object['values'](data.results).forEach((queryRes: any) => { if (!queryRes.series) { return; } const unit = this.resolvePanelUnitFromTargets(options.targets); queryRes.series.forEach((series: any) => { let timeSerie: any = { target: series.name, datapoints: series.points, refId: queryRes.refId, meta: queryRes.meta, }; if (unit) { timeSerie = { ...timeSerie, unit }; } result.push(timeSerie); }); }); return { data: result }; } else { return { data: [] }; } } async annotationQuery(options: any) { const annotation = options.annotation; const queries = [ { refId: 'annotationQuery', datasourceId: this.id, metricType: this.templateSrv.replace(annotation.target.metricType, options.scopedVars || {}), crossSeriesReducer: 'REDUCE_NONE', perSeriesAligner: 'ALIGN_NONE', title: this.templateSrv.replace(annotation.target.title, options.scopedVars || {}), text: this.templateSrv.replace(annotation.target.text, options.scopedVars || {}), tags: this.templateSrv.replace(annotation.target.tags, options.scopedVars || {}), view: 'FULL', filters: this.interpolateFilters(annotation.target.filters || [], options.scopedVars), type: 'annotationQuery', }, ]; const { data } = await this.backendSrv.datasourceRequest({ url: '/api/tsdb/query', method: 'POST', data: { from: options.range.from.valueOf().toString(), to: options.range.to.valueOf().toString(), queries, }, }); const results = data.results['annotationQuery'].tables[0].rows.map((v: any) => { return { annotation: annotation, time: Date.parse(v[0]), title: v[1], tags: [], text: v[3], } as any; }); return results; } async metricFindQuery(query: string) { const stackdriverMetricFindQuery = new StackdriverMetricFindQuery(this); return stackdriverMetricFindQuery.execute(query); } async testDatasource() { let status, message; const defaultErrorMessage = 'Cannot connect to Stackdriver API'; try { const projectName = await this.getDefaultProject(); const path = `v3/projects/${projectName}/metricDescriptors`; const response = await this.doRequest(`${this.baseUrl}${path}`); if (response.status === 200) { status = 'success'; message = 'Successfully queried the Stackdriver API.'; } else { status = 'error'; message = response.statusText ? response.statusText : defaultErrorMessage; } } catch (error) { status = 'error'; if (_.isString(error)) { message = error; } else { message = 'Stackdriver: '; message += error.statusText ? error.statusText : defaultErrorMessage; if (error.data && error.data.error && error.data.error.code) { message += ': ' + error.data.error.code + '. ' + error.data.error.message; } } } finally { return { status, message, }; } } formatStackdriverError(error: any) { let message = 'Stackdriver: '; message += error.statusText ? error.statusText + ': ' : ''; if (error.data && error.data.error) { try { const res = JSON.parse(error.data.error); message += res.error.code + '. ' + res.error.message; } catch (err) { message += error.data.error; } } else { message += 'Cannot connect to Stackdriver API'; } return message; } async getDefaultProject() { try { if (this.authenticationType === 'gce' || !this.projectName) { const { data } = await this.backendSrv.datasourceRequest({ url: '/api/tsdb/query', method: 'POST', data: { queries: [ { refId: 'ensureDefaultProjectQuery', type: 'ensureDefaultProjectQuery', datasourceId: this.id, }, ], }, }); this.projectName = data.results.ensureDefaultProjectQuery.meta.defaultProject; return this.projectName; } else { return this.projectName; } } catch (error) { throw this.formatStackdriverError(error); } } async getMetricTypes(projectName: string): Promise<MetricDescriptor[]> { try { if (this.metricTypes.length === 0) { const metricsApiPath = `v3/projects/${projectName}/metricDescriptors`; const { data } = await this.doRequest(`${this.baseUrl}${metricsApiPath}`); this.metricTypes = data.metricDescriptors.map((m: any) => { const [service] = m.type.split('/'); const [serviceShortName] = service.split('.'); m.service = service; m.serviceShortName = serviceShortName; m.displayName = m.displayName || m.type; return m; }); } return this.metricTypes; } catch (error) { appEvents.emit(CoreEvents.dsRequestError, { error: { data: { error: this.formatStackdriverError(error) } } }); return []; } } async doRequest(url: string, maxRetries = 1) { return this.backendSrv .datasourceRequest({ url: this.url + url, method: 'GET', }) .catch((error: any) => { if (maxRetries > 0) { return this.doRequest(url, maxRetries - 1); } throw error; }); } }