Skip to content

Commit e508db9

Browse files
committed
OpenTSDB: Support for template variable values lookup queries, Closes grafana#1250
1 parent f7b7401 commit e508db9

File tree

5 files changed

+115
-37
lines changed

5 files changed

+115
-37
lines changed

CHANGELOG.md

+1
Original file line numberDiff line numberDiff line change
@@ -4,6 +4,7 @@
44
- [Issue #1525](https://github.com/grafana/grafana/issues/1525). InfluxDB: Full support for InfluxDB 0.9 with new adapted query editor
55
- [Issue #2191](https://github.com/grafana/grafana/issues/2191). KariosDB: Grafana now ships with a KariosDB data source plugin, thx @masaori335
66
- [Issue #1177](https://github.com/grafana/grafana/issues/1177). OpenTSDB: Limit tags by metric, OpenTSDB config option tsd.core.meta.enable_realtime_ts must enabled for OpenTSDB lookup api
7+
- [Issue #1250](https://github.com/grafana/grafana/issues/1250). OpenTSDB: Support for template variable values lookup queries
78

89
**New dashboard features**
910
- [Issue #1144](https://github.com/grafana/grafana/issues/1144). Templating: You can now select multiple template variables values at the same time.

public/app/plugins/datasource/opentsdb/datasource.js

+51-31
Original file line numberDiff line numberDiff line change
@@ -76,35 +76,20 @@ function (angular, _, kbn) {
7676
return backendSrv.datasourceRequest(options);
7777
};
7878

79-
OpenTSDBDatasource.prototype.performSuggestQuery = function(query, type) {
80-
var options = {
81-
method: 'GET',
82-
url: this.url + '/api/suggest',
83-
params: {
84-
type: type,
85-
q: query
86-
}
87-
};
88-
return backendSrv.datasourceRequest(options).then(function(result) {
79+
OpenTSDBDatasource.prototype._performSuggestQuery = function(query) {
80+
return this._get('/api/suggest', {type: 'metrics', q: query}).then(function(result) {
8981
return result.data;
9082
});
9183
};
9284

93-
OpenTSDBDatasource.prototype.performMetricKeyValueLookup = function(metric, key) {
85+
OpenTSDBDatasource.prototype._performMetricKeyValueLookup = function(metric, key) {
9486
if(!metric || !key) {
9587
return $q.when([]);
9688
}
9789

9890
var m = metric + "{" + key + "=*}";
99-
var options = {
100-
method: 'GET',
101-
url: this.url + '/api/search/lookup',
102-
params: {
103-
m: m,
104-
}
105-
};
10691

107-
return backendSrv.datasourceRequest(options).then(function(result) {
92+
return this._get('/api/search/lookup', {m: m}).then(function(result) {
10893
result = result.data.results;
10994
var tagvs = [];
11095
_.each(result, function(r) {
@@ -114,18 +99,10 @@ function (angular, _, kbn) {
11499
});
115100
};
116101

117-
OpenTSDBDatasource.prototype.performMetricKeyLookup = function(metric) {
118-
if(metric === "") {
119-
throw "Metric not set.";
120-
}
121-
var options = {
122-
method: 'GET',
123-
url: this.url + '/api/search/lookup',
124-
params: {
125-
m: metric,
126-
}
127-
};
128-
return backendSrv.datasourceRequest(options).then(function(result) {
102+
OpenTSDBDatasource.prototype._performMetricKeyLookup = function(metric) {
103+
if(!metric) { return $q.when([]); }
104+
105+
return this._get('/api/search/lookup', {m: metric}).then(function(result) {
129106
result = result.data.results;
130107
var tagks = [];
131108
_.each(result, function(r) {
@@ -139,6 +116,49 @@ function (angular, _, kbn) {
139116
});
140117
};
141118

119+
OpenTSDBDatasource.prototype._get = function(relativeUrl, params) {
120+
return backendSrv.datasourceRequest({
121+
method: 'GET',
122+
url: this.url + relativeUrl,
123+
params: params,
124+
});
125+
};
126+
127+
OpenTSDBDatasource.prototype.metricFindQuery = function(query) {
128+
var interpolated;
129+
try {
130+
interpolated = templateSrv.replace(query);
131+
}
132+
catch (err) {
133+
return $q.reject(err);
134+
}
135+
136+
var responseTransform = function(result) {
137+
return _.map(result, function(value) {
138+
return {text: value};
139+
});
140+
};
141+
142+
var metrics_regex = /metrics\((.*)\)/;
143+
var tag_names_regex = /tag_names\((.*)\)/;
144+
var tag_values_regex = /tag_values\((\w+),\s?(\w+)/;
145+
146+
var metrics_query = interpolated.match(metrics_regex);
147+
if (metrics_query) {
148+
return this._performSuggestQuery(metrics_query[1]).then(responseTransform);
149+
}
150+
var tag_names_query = interpolated.match(tag_names_regex);
151+
152+
if (tag_names_query) {
153+
return this._performMetricKeyLookup(tag_names_query[1]).then(responseTransform);
154+
}
155+
156+
var tag_values_query = interpolated.match(tag_values_regex);
157+
if (tag_values_query) {
158+
return this._performMetricKeyValueLookup(tag_values_query[1], tag_values_query[2]).then(responseTransform);
159+
}
160+
};
161+
142162
OpenTSDBDatasource.prototype.testDatasource = function() {
143163
return this.performSuggestQuery('cpu', 'metrics').then(function () {
144164
return { status: "success", message: "Data source is working", title: "Success" };

public/app/plugins/datasource/opentsdb/queryCtrl.js

+10-6
Original file line numberDiff line numberDiff line change
@@ -42,21 +42,25 @@ function (angular, _, kbn) {
4242
$scope.panel.targets.push(clone);
4343
};
4444

45+
$scope.getTextValues = function(metricFindResult) {
46+
return _.map(metricFindResult, function(value) { return value.text; });
47+
};
48+
4549
$scope.suggestMetrics = function(query, callback) {
46-
$scope.datasource
47-
.performSuggestQuery(query, 'metrics')
50+
$scope.datasource.metricFindQuery('metrics()')
51+
.then($scope.getTextValues)
4852
.then(callback);
4953
};
5054

5155
$scope.suggestTagKeys = function(query, callback) {
52-
$scope.datasource
53-
.performMetricKeyLookup($scope.target.metric)
56+
$scope.datasource.metricFindQuery('tag_names(' + $scope.target.metric + ')')
57+
.then($scope.getTextValues)
5458
.then(callback);
5559
};
5660

5761
$scope.suggestTagValues = function(query, callback) {
58-
$scope.datasource
59-
.performMetricKeyValueLookup($scope.target.metric, $scope.target.currentTagKey)
62+
$scope.datasource.metricFindQuery('tag_names(' + $scope.target.metric + ',' + $scope.target.currentTagKey + ')')
63+
.then($scope.getTextValues)
6064
.then(callback);
6165
};
6266

Original file line numberDiff line numberDiff line change
@@ -0,0 +1,52 @@
1+
define([
2+
'helpers',
3+
'plugins/datasource/opentsdb/datasource'
4+
], function(helpers) {
5+
'use strict';
6+
7+
describe('opentsdb', function() {
8+
var ctx = new helpers.ServiceTestContext();
9+
10+
beforeEach(module('grafana.services'));
11+
beforeEach(ctx.providePhase(['backendSrv']));
12+
13+
beforeEach(ctx.createService('OpenTSDBDatasource'));
14+
beforeEach(function() {
15+
ctx.ds = new ctx.service({ url: [''] });
16+
});
17+
18+
describe('When performing metricFindQuery', function() {
19+
var results;
20+
var requestOptions;
21+
22+
beforeEach(function() {
23+
ctx.backendSrv.datasourceRequest = function(options) {
24+
requestOptions = options;
25+
return ctx.$q.when({data: [{ target: 'prod1.count', datapoints: [[10, 1], [12,1]] }]});
26+
};
27+
});
28+
29+
it('metrics() should generate api suggest query', function() {
30+
ctx.ds.metricFindQuery('metrics()').then(function(data) { results = data; });
31+
ctx.$rootScope.$apply();
32+
expect(requestOptions.url).to.be('/api/suggest');
33+
});
34+
35+
it('tag_names(cpu) should generate looku query', function() {
36+
ctx.ds.metricFindQuery('tag_names(cpu)').then(function(data) { results = data; });
37+
ctx.$rootScope.$apply();
38+
expect(requestOptions.url).to.be('/api/search/lookup');
39+
expect(requestOptions.params.m).to.be('cpu');
40+
});
41+
42+
it('tag_values(cpu, test) should generate looku query', function() {
43+
ctx.ds.metricFindQuery('tag_values(cpu, hostname)').then(function(data) { results = data; });
44+
ctx.$rootScope.$apply();
45+
expect(requestOptions.url).to.be('/api/search/lookup');
46+
expect(requestOptions.params.m).to.be('cpu{hostname=*}');
47+
});
48+
49+
});
50+
});
51+
});
52+

public/test/test-main.js

+1
Original file line numberDiff line numberDiff line change
@@ -146,6 +146,7 @@ require([
146146
'specs/dynamicDashboardSrv-specs',
147147
'specs/unsavedChangesSrv-specs',
148148
'specs/valueSelectDropdown-specs',
149+
'specs/opentsdbDatasource-specs',
149150
];
150151

151152
var pluginSpecs = (config.plugins.specs || []).map(function (spec) {

0 commit comments

Comments
 (0)