| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291 |
- ///<reference path="../../../headers/common.d.ts" />
- import angular from 'angular';
- import _ from 'lodash';
- import moment from 'moment';
- import * as dateMath from 'app/core/utils/datemath';
- /** @ngInject */
- export function GraphiteDatasource(instanceSettings, $q, backendSrv, templateSrv) {
- this.basicAuth = instanceSettings.basicAuth;
- this.url = instanceSettings.url;
- this.name = instanceSettings.name;
- this.graphiteVersion = instanceSettings.jsonData.graphiteVersion || '0.9';
- this.cacheTimeout = instanceSettings.cacheTimeout;
- this.withCredentials = instanceSettings.withCredentials;
- this.render_method = instanceSettings.render_method || 'POST';
- this.getQueryOptionsInfo = function() {
- return {
- "maxDataPoints": true,
- "cacheTimeout": true,
- "links": [
- {
- text: "Help",
- url: "http://docs.grafana.org/features/datasources/graphite/#using-graphite-in-grafana"
- }
- ]
- };
- };
- this.query = function(options) {
- var graphOptions = {
- from: this.translateTime(options.rangeRaw.from, false),
- until: this.translateTime(options.rangeRaw.to, true),
- targets: options.targets,
- format: options.format,
- cacheTimeout: options.cacheTimeout || this.cacheTimeout,
- maxDataPoints: options.maxDataPoints,
- };
- var params = this.buildGraphiteParams(graphOptions, options.scopedVars);
- if (params.length === 0) {
- return $q.when({data: []});
- }
- var httpOptions: any = {
- method: 'POST',
- url: '/render',
- data: params.join('&'),
- headers: {
- 'Content-Type': 'application/x-www-form-urlencoded'
- },
- };
- if (options.panelId) {
- httpOptions.requestId = this.name + '.panelId.' + options.panelId;
- }
- return this.doGraphiteRequest(httpOptions).then(this.convertDataPointsToMs);
- };
- this.convertDataPointsToMs = function(result) {
- if (!result || !result.data) { return []; }
- for (var i = 0; i < result.data.length; i++) {
- var series = result.data[i];
- for (var y = 0; y < series.datapoints.length; y++) {
- series.datapoints[y][1] *= 1000;
- }
- }
- return result;
- };
- this.annotationQuery = function(options) {
- // Graphite metric as annotation
- if (options.annotation.target) {
- var target = templateSrv.replace(options.annotation.target, {}, 'glob');
- var graphiteQuery = {
- rangeRaw: options.rangeRaw,
- targets: [{ target: target }],
- format: 'json',
- maxDataPoints: 100
- };
- return this.query(graphiteQuery).then(function(result) {
- var list = [];
- for (var i = 0; i < result.data.length; i++) {
- var target = result.data[i];
- for (var y = 0; y < target.datapoints.length; y++) {
- var datapoint = target.datapoints[y];
- if (!datapoint[0]) { continue; }
- list.push({
- annotation: options.annotation,
- time: datapoint[1],
- title: target.target
- });
- }
- }
- return list;
- });
- } else {
- // Graphite event as annotation
- var tags = templateSrv.replace(options.annotation.tags);
- return this.events({range: options.rangeRaw, tags: tags}).then(function(results) {
- var list = [];
- for (var i = 0; i < results.data.length; i++) {
- var e = results.data[i];
- list.push({
- annotation: options.annotation,
- time: e.when * 1000,
- title: e.what,
- tags: e.tags,
- text: e.data
- });
- }
- return list;
- });
- }
- };
- this.events = function(options) {
- try {
- var tags = '';
- if (options.tags) {
- tags = '&tags=' + options.tags;
- }
- return this.doGraphiteRequest({
- method: 'GET',
- url: '/events/get_data?from=' + this.translateTime(options.range.from, false) +
- '&until=' + this.translateTime(options.range.to, true) + tags,
- });
- } catch (err) {
- return $q.reject(err);
- }
- };
- this.targetContainsTemplate = function(target) {
- return templateSrv.variableExists(target.target);
- };
- this.translateTime = function(date, roundUp) {
- if (_.isString(date)) {
- if (date === 'now') {
- return 'now';
- } else if (date.indexOf('now-') >= 0 && date.indexOf('/') === -1) {
- date = date.substring(3);
- date = date.replace('m', 'min');
- date = date.replace('M', 'mon');
- return date;
- }
- date = dateMath.parse(date, roundUp);
- }
- // graphite' s from filter is exclusive
- // here we step back one minute in order
- // to guarantee that we get all the data that
- // exists for the specified range
- if (roundUp) {
- if (date.get('s')) {
- date.add(1, 'm');
- }
- } else if (roundUp === false) {
- if (date.get('s')) {
- date.subtract(1, 'm');
- }
- }
- return date.unix();
- };
- this.metricFindQuery = function(query, optionalOptions) {
- let options = optionalOptions || {};
- let interpolatedQuery = templateSrv.replace(query);
- let httpOptions: any = {
- method: 'GET',
- url: '/metrics/find',
- params: {
- query: interpolatedQuery
- },
- // for cancellations
- requestId: options.requestId,
- };
- if (options && options.range) {
- httpOptions.params.from = this.translateTime(options.range.from, false);
- httpOptions.params.until = this.translateTime(options.range.to, true);
- }
- return this.doGraphiteRequest(httpOptions).then(results => {
- return _.map(results.data, metric => {
- return {
- text: metric.text,
- expandable: metric.expandable ? true : false
- };
- });
- });
- };
- this.testDatasource = function() {
- return this.metricFindQuery('*').then(function () {
- return { status: "success", message: "Data source is working", title: "Success" };
- });
- };
- this.doGraphiteRequest = function(options) {
- if (this.basicAuth || this.withCredentials) {
- options.withCredentials = true;
- }
- if (this.basicAuth) {
- options.headers = options.headers || {};
- options.headers.Authorization = this.basicAuth;
- }
- options.url = this.url + options.url;
- options.inspect = {type: 'graphite'};
- return backendSrv.datasourceRequest(options);
- };
- this._seriesRefLetters = 'ABCDEFGHIJKLMNOPQRSTUVWXYZ';
- this.buildGraphiteParams = function(options, scopedVars) {
- var graphite_options = ['from', 'until', 'rawData', 'format', 'maxDataPoints', 'cacheTimeout'];
- var clean_options = [], targets = {};
- var target, targetValue, i;
- var regex = /\#([A-Z])/g;
- var intervalFormatFixRegex = /'(\d+)m'/gi;
- var hasTargets = false;
- options['format'] = 'json';
- function fixIntervalFormat(match) {
- return match.replace('m', 'min').replace('M', 'mon');
- }
- for (i = 0; i < options.targets.length; i++) {
- target = options.targets[i];
- if (!target.target) {
- continue;
- }
- if (!target.refId) {
- target.refId = this._seriesRefLetters[i];
- }
- targetValue = templateSrv.replace(target.target, scopedVars);
- targetValue = targetValue.replace(intervalFormatFixRegex, fixIntervalFormat);
- targets[target.refId] = targetValue;
- }
- function nestedSeriesRegexReplacer(match, g1) {
- return targets[g1] || match;
- }
- for (i = 0; i < options.targets.length; i++) {
- target = options.targets[i];
- if (!target.target) {
- continue;
- }
- targetValue = targets[target.refId];
- targetValue = targetValue.replace(regex, nestedSeriesRegexReplacer);
- targets[target.refId] = targetValue;
- if (!target.hide) {
- hasTargets = true;
- clean_options.push("target=" + encodeURIComponent(targetValue));
- }
- }
- _.each(options, function (value, key) {
- if (_.indexOf(graphite_options, key) === -1) { return; }
- if (value) {
- clean_options.push(key + "=" + encodeURIComponent(value));
- }
- });
- if (!hasTargets) {
- return [];
- }
- return clean_options;
- };
- }
|