bucket_agg.js 7.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235
  1. define([
  2. 'angular',
  3. 'lodash',
  4. './query_def',
  5. ],
  6. function (angular, _, queryDef) {
  7. 'use strict';
  8. var module = angular.module('grafana.directives');
  9. module.directive('elasticBucketAgg', function() {
  10. return {
  11. templateUrl: 'public/app/plugins/datasource/elasticsearch/partials/bucket_agg.html',
  12. controller: 'ElasticBucketAggCtrl',
  13. restrict: 'E',
  14. scope: {
  15. target: "=",
  16. index: "=",
  17. onChange: "&",
  18. // getNestedKeys: "&",
  19. getFields: "&",
  20. }
  21. };
  22. });
  23. module.controller('ElasticBucketAggCtrl', function($scope, uiSegmentSrv, $q, $rootScope) {
  24. var bucketAggs = $scope.target.bucketAggs;
  25. $scope.orderByOptions = [];
  26. $scope.bucketAggTypes = queryDef.bucketAggTypes;
  27. $scope.orderOptions = queryDef.orderOptions;
  28. $scope.sizeOptions = queryDef.sizeOptions;
  29. $rootScope.onAppEvent('elastic-query-updated', function() {
  30. $scope.validateModel();
  31. $scope.updateOrderByOptions();
  32. }, $scope);
  33. $scope.init = function() {
  34. $scope.agg = bucketAggs[$scope.index];
  35. $scope.validateModel();
  36. };
  37. $scope.onChangeInternal = function() {
  38. $scope.onChange();
  39. };
  40. $scope.onTypeChanged = function() {
  41. $scope.agg.settings = {};
  42. $scope.showOptions = false;
  43. switch($scope.agg.type) {
  44. case 'date_histogram':
  45. case 'terms': {
  46. delete $scope.agg.query;
  47. delete $scope.agg.settings.nested.path;
  48. $scope.agg.field = 'select field';
  49. break;
  50. }
  51. case 'filters': {
  52. delete $scope.agg.field;
  53. delete $scope.agg.settings.nested.path;
  54. $scope.agg.query = '*';
  55. break;
  56. }
  57. case 'nested': {
  58. delete $scope.agg.field;
  59. delete $scope.agg.query;
  60. $scope.agg.settings.nested = {};
  61. $scope.agg.settings.nested.path = 'select field (type: nested)';
  62. $scope.agg.settings.nested.term = 'select nested term path';
  63. $scope.agg.settings.nested.query = 'select query for Nested Term';
  64. break;
  65. }
  66. case 'geohash_grid': {
  67. $scope.agg.settings.precision = 3;
  68. break;
  69. }
  70. }
  71. $scope.validateModel();
  72. $scope.onChange();
  73. };
  74. $scope.validateModel = function() {
  75. $scope.index = _.indexOf(bucketAggs, $scope.agg);
  76. $scope.isFirst = $scope.index === 0;
  77. $scope.bucketAggCount = bucketAggs.length;
  78. var settingsLinkText = "";
  79. var settings = $scope.agg.settings || {};
  80. switch($scope.agg.type) {
  81. case 'nested': {
  82. if (settingsLinkText === '') {
  83. settingsLinkText = 'Options';
  84. }
  85. break;
  86. }
  87. case 'terms': {
  88. settings.order = settings.order || "asc";
  89. settings.size = settings.size || "10";
  90. settings.min_doc_count = settings.min_doc_count || 0;
  91. settings.orderBy = settings.orderBy || "_term";
  92. if (settings.size !== '0') {
  93. settingsLinkText = queryDef.describeOrder(settings.order) + ' ' + settings.size + ', ';
  94. }
  95. if (settings.min_doc_count > 0) {
  96. settingsLinkText += 'Min Doc Count: ' + settings.min_doc_count + ', ';
  97. }
  98. settingsLinkText += 'Order by: ' + queryDef.describeOrderBy(settings.orderBy, $scope.target);
  99. if (settings.size === '0') {
  100. settingsLinkText += ' (' + settings.order + ')';
  101. }
  102. break;
  103. }
  104. case 'filters': {
  105. settings.filters = settings.filters || [{query: '*'}];
  106. settingsLinkText = _.reduce(settings.filters, function(memo, value, index) {
  107. memo += 'Q' + (index + 1) + ' = ' + value.query + ' ';
  108. return memo;
  109. }, '');
  110. if (settingsLinkText.length > 50) {
  111. settingsLinkText = settingsLinkText.substr(0, 50) + "...";
  112. }
  113. settingsLinkText = 'Filter Queries (' + settings.filters.length + ')';
  114. break;
  115. }
  116. case 'date_histogram': {
  117. settings.interval = settings.interval || 'auto';
  118. settings.min_doc_count = settings.min_doc_count || 0;
  119. $scope.agg.field = $scope.target.timeField;
  120. settingsLinkText = 'Interval: ' + settings.interval;
  121. if (settings.min_doc_count > 0) {
  122. settingsLinkText += ', Min Doc Count: ' + settings.min_doc_count;
  123. }
  124. if (settings.trimEdges === undefined || settings.trimEdges < 0) {
  125. settings.trimEdges = 0;
  126. }
  127. if (settings.trimEdges && settings.trimEdges > 0) {
  128. settingsLinkText += ', Trim edges: ' + settings.trimEdges;
  129. }
  130. break;
  131. }
  132. case 'geohash_grid': {
  133. // limit precision to 7
  134. settings.precision = Math.max(Math.min(settings.precision, 7), 1);
  135. settingsLinkText = 'Precision: ' + settings.precision;
  136. break;
  137. }
  138. }
  139. $scope.settingsLinkText = settingsLinkText;
  140. $scope.agg.settings = settings;
  141. return true;
  142. };
  143. $scope.addFiltersQuery = function() {
  144. $scope.agg.settings.filters.push({query: '*'});
  145. };
  146. $scope.removeFiltersQuery = function(filter) {
  147. $scope.agg.settings.filters = _.without($scope.agg.settings.filters, filter);
  148. };
  149. $scope.toggleOptions = function() {
  150. $scope.showOptions = !$scope.showOptions;
  151. $scope.updateOrderByOptions();
  152. };
  153. $scope.updateOrderByOptions = function() {
  154. $scope.orderByOptions = queryDef.getOrderByOptions($scope.target);
  155. };
  156. $scope.getFieldsInternal = function() {
  157. if ($scope.agg.type === 'date_histogram') {
  158. return $scope.getFields({$fieldType: 'date'});
  159. } else {
  160. return $scope.getFields();
  161. }
  162. };
  163. $scope.getFieldsNestedPath = function() {
  164. return $scope.getFields({$fieldType: 'nested'});
  165. };
  166. $scope.getFieldsNestedTerm = function() {
  167. return $scope.getFields({$fieldType: 'string'});
  168. };
  169. // FIX THIS: add a method getNestedKeys to be called here
  170. // for getting nested key string ids, based on term/path supplied.
  171. // $scope.getFieldsNestedQuery = function() {
  172. // return $scope.getNestedKeys();
  173. // };
  174. $scope.getIntervalOptions = function() {
  175. return $q.when(uiSegmentSrv.transformToSegments(true, 'interval')(queryDef.intervalOptions));
  176. };
  177. $scope.addBucketAgg = function() {
  178. // if last is date histogram add it before
  179. var lastBucket = bucketAggs[bucketAggs.length - 1];
  180. var addIndex = bucketAggs.length - 1;
  181. if (lastBucket && lastBucket.type === 'date_histogram') {
  182. addIndex - 1;
  183. }
  184. var id = _.reduce($scope.target.bucketAggs.concat($scope.target.metrics), function(max, val) {
  185. return parseInt(val.id) > max ? parseInt(val.id) : max;
  186. }, 0);
  187. bucketAggs.splice(addIndex, 0, {type: "terms", field: "select field", id: (id+1).toString(), fake: true});
  188. $scope.onChange();
  189. };
  190. $scope.removeBucketAgg = function() {
  191. bucketAggs.splice($scope.index, 1);
  192. $scope.onChange();
  193. };
  194. $scope.init();
  195. });
  196. });