bucket_agg.ts 6.5 KB

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