bucket_agg.ts 6.5 KB

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