bucket_agg.ts 6.4 KB

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