datasource.js 6.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220
  1. define([
  2. 'angular',
  3. 'lodash',
  4. 'app/core/utils/datemath',
  5. './influx_series',
  6. './influx_query',
  7. ],
  8. function (angular, _, dateMath, InfluxSeries, InfluxQuery) {
  9. 'use strict';
  10. InfluxQuery = InfluxQuery.default;
  11. /** @ngInject */
  12. function InfluxDatasource(instanceSettings, $q, backendSrv, templateSrv) {
  13. this.type = 'influxdb';
  14. this.urls = _.map(instanceSettings.url.split(','), function(url) {
  15. return url.trim();
  16. });
  17. this.username = instanceSettings.username;
  18. this.password = instanceSettings.password;
  19. this.name = instanceSettings.name;
  20. this.database = instanceSettings.database;
  21. this.basicAuth = instanceSettings.basicAuth;
  22. this.supportAnnotations = true;
  23. this.supportMetrics = true;
  24. this.query = function(options) {
  25. var timeFilter = getTimeFilter(options);
  26. var queryTargets = [];
  27. var i, y;
  28. var allQueries = _.map(options.targets, function(target) {
  29. if (target.hide) { return []; }
  30. queryTargets.push(target);
  31. // build query
  32. var queryModel = new InfluxQuery(target);
  33. var query = queryModel.render();
  34. query = query.replace(/\$interval/g, (target.interval || options.interval));
  35. return query;
  36. }).join("\n");
  37. // replace grafana variables
  38. allQueries = allQueries.replace(/\$timeFilter/g, timeFilter);
  39. // replace templated variables
  40. allQueries = templateSrv.replace(allQueries, options.scopedVars);
  41. return this._seriesQuery(allQueries).then(function(data) {
  42. if (!data || !data.results) {
  43. return [];
  44. }
  45. var seriesList = [];
  46. for (i = 0; i < data.results.length; i++) {
  47. var result = data.results[i];
  48. if (!result || !result.series) { continue; }
  49. var target = queryTargets[i];
  50. var alias = target.alias;
  51. if (alias) {
  52. alias = templateSrv.replace(target.alias, options.scopedVars);
  53. }
  54. var influxSeries = new InfluxSeries({ series: data.results[i].series, alias: alias });
  55. switch(target.resultFormat) {
  56. case 'table': {
  57. seriesList.push(influxSeries.getTable());
  58. break;
  59. }
  60. default: {
  61. var timeSeries = influxSeries.getTimeSeries();
  62. for (y = 0; y < timeSeries.length; y++) {
  63. seriesList.push(timeSeries[y]);
  64. }
  65. break;
  66. }
  67. }
  68. }
  69. return { data: seriesList };
  70. });
  71. };
  72. this.annotationQuery = function(options) {
  73. var timeFilter = getTimeFilter({rangeRaw: options.rangeRaw});
  74. var query = options.annotation.query.replace('$timeFilter', timeFilter);
  75. query = templateSrv.replace(query);
  76. return this._seriesQuery(query).then(function(data) {
  77. if (!data || !data.results || !data.results[0]) {
  78. throw { message: 'No results in response from InfluxDB' };
  79. }
  80. return new InfluxSeries({series: data.results[0].series, annotation: options.annotation}).getAnnotations();
  81. });
  82. };
  83. this.metricFindQuery = function (query) {
  84. var interpolated;
  85. try {
  86. interpolated = templateSrv.replace(query);
  87. }
  88. catch (err) {
  89. return $q.reject(err);
  90. }
  91. return this._seriesQuery(interpolated).then(function (results) {
  92. if (!results || results.results.length === 0) { return []; }
  93. var influxResults = results.results[0];
  94. if (!influxResults.series) {
  95. return [];
  96. }
  97. var series = influxResults.series[0];
  98. return _.map(series.values, function(value) {
  99. if (_.isArray(value)) {
  100. return { text: value[0] };
  101. } else {
  102. return { text: value };
  103. }
  104. });
  105. });
  106. };
  107. this._seriesQuery = function(query) {
  108. return this._influxRequest('GET', '/query', {q: query, epoch: 'ms'});
  109. };
  110. this.testDatasource = function() {
  111. return this.metricFindQuery('SHOW MEASUREMENTS LIMIT 1').then(function () {
  112. return { status: "success", message: "Data source is working", title: "Success" };
  113. });
  114. };
  115. this._influxRequest = function(method, url, data) {
  116. var self = this;
  117. var currentUrl = self.urls.shift();
  118. self.urls.push(currentUrl);
  119. var params = {
  120. u: self.username,
  121. p: self.password,
  122. };
  123. if (self.database) {
  124. params.db = self.database;
  125. }
  126. if (method === 'GET') {
  127. _.extend(params, data);
  128. data = null;
  129. }
  130. var options = {
  131. method: method,
  132. url: currentUrl + url,
  133. params: params,
  134. data: data,
  135. precision: "ms",
  136. inspect: { type: 'influxdb' },
  137. };
  138. options.headers = options.headers || {};
  139. if (self.basicAuth) {
  140. options.headers.Authorization = self.basicAuth;
  141. }
  142. return backendSrv.datasourceRequest(options).then(function(result) {
  143. return result.data;
  144. }, function(err) {
  145. if (err.status !== 0 || err.status >= 300) {
  146. if (err.data && err.data.error) {
  147. throw { message: 'InfluxDB Error Response: ' + err.data.error, data: err.data, config: err.config };
  148. }
  149. else {
  150. throw { message: 'InfluxDB Error: ' + err.message, data: err.data, config: err.config };
  151. }
  152. }
  153. });
  154. };
  155. function getTimeFilter(options) {
  156. var from = getInfluxTime(options.rangeRaw.from, false);
  157. var until = getInfluxTime(options.rangeRaw.to, true);
  158. var fromIsAbsolute = from[from.length-1] === 's';
  159. if (until === 'now()' && !fromIsAbsolute) {
  160. return 'time > ' + from;
  161. }
  162. return 'time > ' + from + ' and time < ' + until;
  163. }
  164. function getInfluxTime(date, roundUp) {
  165. if (_.isString(date)) {
  166. if (date === 'now') {
  167. return 'now()';
  168. }
  169. var parts = /^now-(\d+)([d|h|m|s])$/.exec(date);
  170. if (parts) {
  171. var amount = parseInt(parts[1]);
  172. var unit = parts[2];
  173. return 'now() - ' + amount + unit;
  174. }
  175. date = dateMath.parse(date, roundUp);
  176. }
  177. return (date.valueOf() / 1000).toFixed(0) + 's';
  178. }
  179. }
  180. return InfluxDatasource;
  181. });