datasource.ts 5.9 KB

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