query_builder.js 4.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169
  1. define([
  2. 'lodash'
  3. ],
  4. function (_) {
  5. 'use strict';
  6. function InfluxQueryBuilder(target, queryModel) {
  7. this.target = target;
  8. this.model = queryModel;
  9. if (target.groupByTags) {
  10. target.groupBy = [{type: 'time', interval: 'auto'}];
  11. for (var i in target.groupByTags) {
  12. target.groupBy.push({type: 'tag', key: target.groupByTags[i]});
  13. }
  14. delete target.groupByTags;
  15. }
  16. }
  17. function renderTagCondition (tag, index) {
  18. var str = "";
  19. var operator = tag.operator;
  20. var value = tag.value;
  21. if (index > 0) {
  22. str = (tag.condition || 'AND') + ' ';
  23. }
  24. if (!operator) {
  25. if (/^\/.*\/$/.test(tag.value)) {
  26. operator = '=~';
  27. } else {
  28. operator = '=';
  29. }
  30. }
  31. // quote value unless regex
  32. if (operator !== '=~' && operator !== '!~') {
  33. value = "'" + value + "'";
  34. }
  35. return str + '"' + tag.key + '" ' + operator + ' ' + value;
  36. }
  37. var p = InfluxQueryBuilder.prototype;
  38. p.build = function() {
  39. return this.target.rawQuery ? this._modifyRawQuery() : this._buildQuery();
  40. };
  41. p.buildExploreQuery = function(type, withKey) {
  42. var query;
  43. var measurement;
  44. if (type === 'TAG_KEYS') {
  45. query = 'SHOW TAG KEYS';
  46. measurement = this.target.measurement;
  47. } else if (type === 'TAG_VALUES') {
  48. query = 'SHOW TAG VALUES';
  49. measurement = this.target.measurement;
  50. } else if (type === 'MEASUREMENTS') {
  51. query = 'SHOW MEASUREMENTS';
  52. } else if (type === 'FIELDS') {
  53. query = 'SHOW FIELD KEYS FROM "' + this.target.measurement + '"';
  54. return query;
  55. }
  56. if (measurement) {
  57. if (!measurement.match('^/.*/') && !measurement.match(/^merge\(.*\)/)) {
  58. measurement = '"' + measurement+ '"';
  59. }
  60. query += ' FROM ' + measurement;
  61. }
  62. if (withKey) {
  63. query += ' WITH KEY = "' + withKey + '"';
  64. }
  65. if (this.target.tags && this.target.tags.length > 0) {
  66. var whereConditions = _.reduce(this.target.tags, function(memo, tag) {
  67. // do not add a condition for the key we want to explore for
  68. if (tag.key === withKey) {
  69. return memo;
  70. }
  71. memo.push(renderTagCondition(tag, memo.length));
  72. return memo;
  73. }, []);
  74. if (whereConditions.length > 0) {
  75. query += ' WHERE ' + whereConditions.join(' ');
  76. }
  77. }
  78. return query;
  79. };
  80. p._getGroupByTimeInterval = function(interval) {
  81. if (interval === 'auto') {
  82. return '$interval';
  83. }
  84. return interval;
  85. };
  86. p._buildQuery = function() {
  87. var target = this.target;
  88. if (!target.measurement) {
  89. throw "Metric measurement is missing";
  90. }
  91. if (!target.fields) {
  92. target.fields = [{name: 'value', func: target.function || 'mean'}];
  93. }
  94. var query = 'SELECT ';
  95. var i;
  96. for (i = 0; i < target.fields.length; i++) {
  97. var field = target.fields[i];
  98. if (i > 0) {
  99. query += ', ';
  100. }
  101. query += field.func + '("' + field.name + '")';
  102. if (field.mathExpr) {
  103. query += field.mathExpr;
  104. }
  105. if (field.asExpr) {
  106. query += ' AS "' + field.asExpr + '"';
  107. } else {
  108. query += ' AS "' + field.name + '"';
  109. }
  110. }
  111. var measurement = target.measurement;
  112. if (!measurement.match('^/.*/') && !measurement.match(/^merge\(.*\)/)) {
  113. measurement = '"' + measurement+ '"';
  114. }
  115. query += ' FROM ' + measurement + ' WHERE ';
  116. var conditions = _.map(target.tags, function(tag, index) {
  117. return renderTagCondition(tag, index);
  118. });
  119. query += conditions.join(' ');
  120. query += (conditions.length > 0 ? ' AND ' : '') + '$timeFilter';
  121. query += ' GROUP BY';
  122. for (i = 0; i < target.groupBy.length; i++) {
  123. var group = target.groupBy[i];
  124. if (group.type === 'time') {
  125. query += ' time(' + this._getGroupByTimeInterval(group.interval) + ')';
  126. } else {
  127. query += ', "' + group.key + '"';
  128. }
  129. }
  130. if (target.fill) {
  131. query += ' fill(' + target.fill + ')';
  132. }
  133. target.query = query;
  134. return query;
  135. };
  136. p._modifyRawQuery = function () {
  137. return this.target.query.replace(";", "");
  138. };
  139. return InfluxQueryBuilder;
  140. });