Commit ee92fee2 by Carl Bergquist

Merge pull request #4460 from grafana/influxdb

[InfluxDB] Adds support for 0.11.0 tags
parents b06d8093 5f2f4a08
...@@ -6,9 +6,23 @@ import _ from 'lodash'; ...@@ -6,9 +6,23 @@ import _ from 'lodash';
import * as dateMath from 'app/core/utils/datemath'; import * as dateMath from 'app/core/utils/datemath';
import InfluxSeries from './influx_series'; import InfluxSeries from './influx_series';
import InfluxQuery from './influx_query'; import InfluxQuery from './influx_query';
import ResponseParser from './response_parser';
/** @ngInject */
export function InfluxDatasource(instanceSettings, $q, backendSrv, templateSrv) { export default class InfluxDatasource {
type: string;
urls: any;
username: string;
password: string;
name: string;
database: any;
basicAuth: any;
interval: any;
supportAnnotations: boolean;
supportMetrics: boolean;
responseParser: any;
/** @ngInject */
constructor(instanceSettings, private $q, private backendSrv, private templateSrv) {
this.type = 'influxdb'; this.type = 'influxdb';
this.urls = _.map(instanceSettings.url.split(','), function(url) { this.urls = _.map(instanceSettings.url.split(','), function(url) {
return url.trim(); return url.trim();
...@@ -22,19 +36,21 @@ export function InfluxDatasource(instanceSettings, $q, backendSrv, templateSrv) ...@@ -22,19 +36,21 @@ export function InfluxDatasource(instanceSettings, $q, backendSrv, templateSrv)
this.interval = (instanceSettings.jsonData || {}).timeInterval; this.interval = (instanceSettings.jsonData || {}).timeInterval;
this.supportAnnotations = true; this.supportAnnotations = true;
this.supportMetrics = true; this.supportMetrics = true;
this.responseParser = new ResponseParser();
}
this.query = function(options) { query(options) {
var timeFilter = getTimeFilter(options); var timeFilter = this.getTimeFilter(options);
var queryTargets = []; var queryTargets = [];
var i, y; var i, y;
var allQueries = _.map(options.targets, function(target) { var allQueries = _.map(options.targets, (target) => {
if (target.hide) { return []; } if (target.hide) { return []; }
queryTargets.push(target); queryTargets.push(target);
// build query // build query
var queryModel = new InfluxQuery(target, templateSrv, options.scopedVars); var queryModel = new InfluxQuery(target, this.templateSrv, options.scopedVars);
var query = queryModel.render(true); var query = queryModel.render(true);
query = query.replace(/\$interval/g, (target.interval || options.interval)); query = query.replace(/\$interval/g, (target.interval || options.interval));
return query; return query;
...@@ -45,9 +61,9 @@ export function InfluxDatasource(instanceSettings, $q, backendSrv, templateSrv) ...@@ -45,9 +61,9 @@ export function InfluxDatasource(instanceSettings, $q, backendSrv, templateSrv)
allQueries = allQueries.replace(/\$timeFilter/g, timeFilter); allQueries = allQueries.replace(/\$timeFilter/g, timeFilter);
// replace templated variables // replace templated variables
allQueries = templateSrv.replace(allQueries, options.scopedVars); allQueries = this.templateSrv.replace(allQueries, options.scopedVars);
return this._seriesQuery(allQueries).then(function(data): any { return this._seriesQuery(allQueries).then((data): any => {
if (!data || !data.results) { if (!data || !data.results) {
return []; return [];
} }
...@@ -60,7 +76,7 @@ export function InfluxDatasource(instanceSettings, $q, backendSrv, templateSrv) ...@@ -60,7 +76,7 @@ export function InfluxDatasource(instanceSettings, $q, backendSrv, templateSrv)
var target = queryTargets[i]; var target = queryTargets[i];
var alias = target.alias; var alias = target.alias;
if (alias) { if (alias) {
alias = templateSrv.replace(target.alias, options.scopedVars); alias = this.templateSrv.replace(target.alias, options.scopedVars);
} }
var influxSeries = new InfluxSeries({ series: data.results[i].series, alias: alias }); var influxSeries = new InfluxSeries({ series: data.results[i].series, alias: alias });
...@@ -84,16 +100,16 @@ export function InfluxDatasource(instanceSettings, $q, backendSrv, templateSrv) ...@@ -84,16 +100,16 @@ export function InfluxDatasource(instanceSettings, $q, backendSrv, templateSrv)
}); });
}; };
this.annotationQuery = function(options) { annotationQuery(options) {
if (!options.annotation.query) { if (!options.annotation.query) {
return $q.reject({message: 'Query missing in annotation definition'}); return this.$q.reject({message: 'Query missing in annotation definition'});
} }
var timeFilter = getTimeFilter({rangeRaw: options.rangeRaw}); var timeFilter = this.getTimeFilter({rangeRaw: options.rangeRaw});
var query = options.annotation.query.replace('$timeFilter', timeFilter); var query = options.annotation.query.replace('$timeFilter', timeFilter);
query = templateSrv.replace(query); query = this.templateSrv.replace(query);
return this._seriesQuery(query).then(function(data) { return this._seriesQuery(query).then(data => {
if (!data || !data.results || !data.results[0]) { if (!data || !data.results || !data.results[0]) {
throw { message: 'No results in response from InfluxDB' }; throw { message: 'No results in response from InfluxDB' };
} }
...@@ -101,44 +117,29 @@ export function InfluxDatasource(instanceSettings, $q, backendSrv, templateSrv) ...@@ -101,44 +117,29 @@ export function InfluxDatasource(instanceSettings, $q, backendSrv, templateSrv)
}); });
}; };
this.metricFindQuery = function (query) { metricFindQuery(query) {
var interpolated; var interpolated;
try { try {
interpolated = templateSrv.replace(query, null, 'regex'); interpolated = this.templateSrv.replace(query, null, 'regex');
} catch (err) { } catch (err) {
return $q.reject(err); return this.$q.reject(err);
} }
return this._seriesQuery(interpolated).then(function (results) { return this._seriesQuery(interpolated)
if (!results || results.results.length === 0) { return []; } .then(_.curry(this.responseParser.parse)(query));
var influxResults = results.results[0];
if (!influxResults.series) {
return [];
}
var series = influxResults.series[0];
return _.map(series.values, function(value) {
if (_.isArray(value)) {
return { text: value[0] };
} else {
return { text: value };
}
});
});
}; };
this._seriesQuery = function(query) { _seriesQuery(query) {
return this._influxRequest('GET', '/query', {q: query, epoch: 'ms'}); return this._influxRequest('GET', '/query', {q: query, epoch: 'ms'});
}; }
this.testDatasource = function() { testDatasource() {
return this.metricFindQuery('SHOW MEASUREMENTS LIMIT 1').then(function () { return this.metricFindQuery('SHOW MEASUREMENTS LIMIT 1').then(() => {
return { status: "success", message: "Data source is working", title: "Success" }; return { status: "success", message: "Data source is working", title: "Success" };
}); });
}; }
this._influxRequest = function(method, url, data) { _influxRequest(method, url, data) {
var self = this; var self = this;
var currentUrl = self.urls.shift(); var currentUrl = self.urls.shift();
...@@ -172,7 +173,7 @@ export function InfluxDatasource(instanceSettings, $q, backendSrv, templateSrv) ...@@ -172,7 +173,7 @@ export function InfluxDatasource(instanceSettings, $q, backendSrv, templateSrv)
options.headers.Authorization = self.basicAuth; options.headers.Authorization = self.basicAuth;
} }
return backendSrv.datasourceRequest(options).then(function(result) { return this.backendSrv.datasourceRequest(options).then(result => {
return result.data; return result.data;
}, function(err) { }, function(err) {
if (err.status !== 0 || err.status >= 300) { if (err.status !== 0 || err.status >= 300) {
...@@ -185,9 +186,9 @@ export function InfluxDatasource(instanceSettings, $q, backendSrv, templateSrv) ...@@ -185,9 +186,9 @@ export function InfluxDatasource(instanceSettings, $q, backendSrv, templateSrv)
}); });
}; };
function getTimeFilter(options) { getTimeFilter(options) {
var from = getInfluxTime(options.rangeRaw.from, false); var from = this.getInfluxTime(options.rangeRaw.from, false);
var until = getInfluxTime(options.rangeRaw.to, true); var until = this.getInfluxTime(options.rangeRaw.to, true);
var fromIsAbsolute = from[from.length-1] === 's'; var fromIsAbsolute = from[from.length-1] === 's';
if (until === 'now()' && !fromIsAbsolute) { if (until === 'now()' && !fromIsAbsolute) {
...@@ -197,7 +198,7 @@ export function InfluxDatasource(instanceSettings, $q, backendSrv, templateSrv) ...@@ -197,7 +198,7 @@ export function InfluxDatasource(instanceSettings, $q, backendSrv, templateSrv)
return 'time > ' + from + ' and time < ' + until; return 'time > ' + from + ' and time < ' + until;
} }
function getInfluxTime(date, roundUp) { getInfluxTime(date, roundUp) {
if (_.isString(date)) { if (_.isString(date)) {
if (date === 'now') { if (date === 'now') {
return 'now()'; return 'now()';
......
...@@ -11,6 +11,7 @@ export default class InfluxQuery { ...@@ -11,6 +11,7 @@ export default class InfluxQuery {
templateSrv: any; templateSrv: any;
scopedVars: any; scopedVars: any;
/** @ngInject */
constructor(target, templateSrv?, scopedVars?) { constructor(target, templateSrv?, scopedVars?) {
this.target = target; this.target = target;
this.templateSrv = templateSrv; this.templateSrv = templateSrv;
......
import {InfluxDatasource} from './datasource'; import InfluxDatasource from './datasource';
import {InfluxQueryCtrl} from './query_ctrl'; import {InfluxQueryCtrl} from './query_ctrl';
class InfluxConfigCtrl { class InfluxConfigCtrl {
......
///<reference path="../../../headers/common.d.ts" />
import _ from 'lodash';
export default class ResponseParser {
parse(query, results) {
if (!results || results.results.length === 0) { return []; }
var influxResults = results.results[0];
if (!influxResults.series) {
return [];
}
var series = influxResults.series[0];
return _.map(series.values, (value) => {
if (_.isArray(value)) {
if (query.indexOf('SHOW TAG VALUES') >= 0) {
return { text: (value[1] || value[0]) };
} else {
return { text: value[0] };
}
} else {
return { text: value };
}
});
}
}
import _ from 'lodash';
import {describe, beforeEach, it, sinon, expect} from 'test/lib/common';
import ResponseParser from '../response_parser';
describe("influxdb response parser", () => {
this.parser = new ResponseParser();
describe("SHOW TAG response", () => {
var query = 'SHOW TAG KEYS FROM "cpu"';
var response = {
"results": [
{
"series": [
{
"name": "cpu",
"columns": ["tagKey"],
"values": [ ["datacenter"], ["hostname"], ["source"] ]
}
]
}
]
};
var result = this.parser.parse(query, response);
it("expects three results", () => {
expect(_.size(result)).to.be(3);
});
});
describe("SHOW TAG VALUES response", () => {
var query = 'SHOW TAG VALUES FROM "cpu" WITH KEY = "hostname"';
describe("response from 0.10.0", () => {
var response = {
"results": [
{
"series": [
{
"name": "hostnameTagValues",
"columns": ["hostname"],
"values": [ ["server1"], ["server2"] ]
}
]
}
]
};
var result = this.parser.parse(query, response);
it("should get two responses", () => {
expect(_.size(result)).to.be(2);
expect(result[0].text).to.be("server1");
expect(result[1].text).to.be("server2");
});
});
describe("response from 0.11.0", () => {
var response = {
"results": [
{
"series": [
{
"name": "cpu",
"columns": [ "key", "value"],
"values": [ [ "source", "site" ], [ "source", "api" ] ]
}
]
}
]
};
var result = this.parser.parse(query, response);
it("should get two responses", () => {
expect(_.size(result)).to.be(2);
expect(result[0].text).to.be('site');
expect(result[1].text).to.be('api');
});
});
});
describe("SHOW FIELD response", () => {
var query = 'SHOW FIELD KEYS FROM "cpu"';
describe("response from 0.10.0", () => {
var response = {
"results": [
{
"series": [
{
"name": "measurements",
"columns": ["name"],
"values": [
["cpu"], ["derivative"], ["logins.count"], ["logs"], ["payment.ended"], ["payment.started"]
]
}
]
}
]
};
var result = this.parser.parse(query, response);
it("should get two responses", () => {
expect(_.size(result)).to.be(6);
});
});
describe("response from 0.11.0", () => {
var response = {
"results": [
{
"series": [
{
"name": "cpu",
"columns": ["fieldKey"],
"values": [ [ "value"] ]
}
]
}
]
};
var result = this.parser.parse(query, response);
it("should get two responses", () => {
expect(_.size(result)).to.be(1);
});
});
});
});
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment