fields.spec.jsx 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343
  1. import {Organization} from 'fixtures/js-stubs/organization';
  2. import {
  3. aggregateMultiPlotType,
  4. aggregateOutputType,
  5. explodeField,
  6. fieldAlignment,
  7. generateAggregateFields,
  8. getAggregateAlias,
  9. isAggregateEquation,
  10. isAggregateField,
  11. isMeasurement,
  12. measurementType,
  13. parseFunction,
  14. } from 'sentry/utils/discover/fields';
  15. describe('parseFunction', function () {
  16. it('returns null on non aggregate fields', function () {
  17. expect(parseFunction('field')).toEqual(null);
  18. expect(parseFunction('under_field')).toEqual(null);
  19. expect(parseFunction('foo.bar.is-Enterprise_42')).toEqual(null);
  20. });
  21. it('handles 0 arg functions', function () {
  22. expect(parseFunction('count()')).toEqual({
  23. name: 'count',
  24. arguments: [],
  25. });
  26. expect(parseFunction('count_unique()')).toEqual({
  27. name: 'count_unique',
  28. arguments: [],
  29. });
  30. });
  31. it('handles 1 arg functions', function () {
  32. expect(parseFunction('count(id)')).toEqual({
  33. name: 'count',
  34. arguments: ['id'],
  35. });
  36. expect(parseFunction('count_unique(user)')).toEqual({
  37. name: 'count_unique',
  38. arguments: ['user'],
  39. });
  40. expect(parseFunction('count_unique(issue.id)')).toEqual({
  41. name: 'count_unique',
  42. arguments: ['issue.id'],
  43. });
  44. expect(parseFunction('count(foo.bar.is-Enterprise_42)')).toEqual({
  45. name: 'count',
  46. arguments: ['foo.bar.is-Enterprise_42'],
  47. });
  48. });
  49. it('handles 2 arg functions', function () {
  50. expect(parseFunction('percentile(transaction.duration,0.81)')).toEqual({
  51. name: 'percentile',
  52. arguments: ['transaction.duration', '0.81'],
  53. });
  54. expect(parseFunction('percentile(transaction.duration, 0.11)')).toEqual({
  55. name: 'percentile',
  56. arguments: ['transaction.duration', '0.11'],
  57. });
  58. });
  59. it('handles 3 arg functions', function () {
  60. expect(parseFunction('count_if(transaction.duration,greater,0.81)')).toEqual({
  61. name: 'count_if',
  62. arguments: ['transaction.duration', 'greater', '0.81'],
  63. });
  64. expect(parseFunction('count_if(some_tag,greater,"0.81,123,152,()")')).toEqual({
  65. name: 'count_if',
  66. arguments: ['some_tag', 'greater', '"0.81,123,152,()"'],
  67. });
  68. expect(parseFunction('function(foo, bar, baz)')).toEqual({
  69. name: 'function',
  70. arguments: ['foo', 'bar', 'baz'],
  71. });
  72. });
  73. it('handles 4 arg functions', function () {
  74. expect(parseFunction('to_other(release,"0.81,123,152,()",others,current)')).toEqual({
  75. name: 'to_other',
  76. arguments: ['release', '"0.81,123,152,()"', 'others', 'current'],
  77. });
  78. });
  79. });
  80. describe('getAggregateAlias', function () {
  81. it('no-ops simple fields', function () {
  82. expect(getAggregateAlias('field')).toEqual('field');
  83. expect(getAggregateAlias('under_field')).toEqual('under_field');
  84. expect(getAggregateAlias('foo.bar.is-Enterprise_42')).toEqual(
  85. 'foo.bar.is-Enterprise_42'
  86. );
  87. });
  88. it('handles 0 arg functions', function () {
  89. expect(getAggregateAlias('count()')).toEqual('count');
  90. expect(getAggregateAlias('count_unique()')).toEqual('count_unique');
  91. });
  92. it('handles 1 arg functions', function () {
  93. expect(getAggregateAlias('count(id)')).toEqual('count_id');
  94. expect(getAggregateAlias('count_unique(user)')).toEqual('count_unique_user');
  95. expect(getAggregateAlias('count_unique(issue.id)')).toEqual('count_unique_issue_id');
  96. expect(getAggregateAlias('count(foo.bar.is-Enterprise_42)')).toEqual(
  97. 'count_foo_bar_is_Enterprise_42'
  98. );
  99. });
  100. it('handles 2 arg functions', function () {
  101. expect(getAggregateAlias('percentile(transaction.duration,0.81)')).toEqual(
  102. 'percentile_transaction_duration_0_81'
  103. );
  104. expect(getAggregateAlias('percentile(transaction.duration, 0.11)')).toEqual(
  105. 'percentile_transaction_duration_0_11'
  106. );
  107. });
  108. it('handles to_other with symbols', function () {
  109. expect(
  110. getAggregateAlias('to_other(release,"release:beta@1.1.1 (2)",others,current)')
  111. ).toEqual('to_other_release__release_beta_1_1_1__2___others_current');
  112. });
  113. });
  114. describe('isAggregateField', function () {
  115. it('detects aliases', function () {
  116. expect(isAggregateField('p888')).toBe(false);
  117. expect(isAggregateField('other_field')).toBe(false);
  118. expect(isAggregateField('foo.bar.is-Enterprise_42')).toBe(false);
  119. });
  120. it('detects functions', function () {
  121. expect(isAggregateField('count()')).toBe(true);
  122. expect(isAggregateField('p75()')).toBe(true);
  123. expect(isAggregateField('percentile(transaction.duration, 0.55)')).toBe(true);
  124. expect(isAggregateField('last_seen()')).toBe(true);
  125. expect(isAggregateField('thing(')).toBe(false);
  126. expect(isAggregateField('unique_count(user)')).toBe(true);
  127. expect(isAggregateField('unique_count(foo.bar.is-Enterprise_42)')).toBe(true);
  128. });
  129. });
  130. describe('isAggregateEquation', function () {
  131. it('detects functions', function () {
  132. expect(isAggregateEquation('equation|5 + count()')).toBe(true);
  133. expect(
  134. isAggregateEquation('equation|percentile(transaction.duration, 0.55) / count()')
  135. ).toBe(true);
  136. expect(isAggregateEquation('equation|(5 + 5) + (count() - 2)')).toBe(true);
  137. });
  138. it('detects lack of functions', function () {
  139. expect(isAggregateEquation('equation|5 + 5')).toBe(false);
  140. expect(isAggregateEquation('equation|(5 + 5)')).toBe(false);
  141. expect(isAggregateEquation('equation|5 + (thing - other_thing)')).toBe(false);
  142. expect(isAggregateEquation('equation|5+(thing-other_thing)')).toBe(false);
  143. });
  144. });
  145. describe('measurement', function () {
  146. it('isMeasurement', function () {
  147. expect(isMeasurement('measurements.fp')).toBe(true);
  148. expect(isMeasurement('measurements.fcp')).toBe(true);
  149. expect(isMeasurement('measurements.lcp')).toBe(true);
  150. expect(isMeasurement('measurements.fid')).toBe(true);
  151. expect(isMeasurement('measurements.foo')).toBe(true);
  152. expect(isMeasurement('measurements.bar')).toBe(true);
  153. expect(isMeasurement('timestamp')).toBe(false);
  154. expect(isMeasurement('project.id')).toBe(false);
  155. expect(isMeasurement('transaction')).toBe(false);
  156. expect(isMeasurement('max(timestamp)')).toBe(false);
  157. expect(isMeasurement('percentile(measurements.fcp, 0.5)')).toBe(false);
  158. });
  159. it('measurementType', function () {
  160. expect(measurementType('measurements.fp')).toBe('duration');
  161. expect(measurementType('measurements.fcp')).toBe('duration');
  162. expect(measurementType('measurements.lcp')).toBe('duration');
  163. expect(measurementType('measurements.fid')).toBe('duration');
  164. expect(measurementType('measurements.foo')).toBe('number');
  165. expect(measurementType('measurements.bar')).toBe('number');
  166. });
  167. });
  168. describe('explodeField', function () {
  169. it('explodes fields', function () {
  170. expect(explodeField({field: 'foobar'})).toEqual({
  171. kind: 'field',
  172. field: 'foobar',
  173. });
  174. // has width
  175. expect(explodeField({field: 'foobar', width: 123})).toEqual({
  176. kind: 'field',
  177. field: 'foobar',
  178. });
  179. // has aggregation
  180. expect(explodeField({field: 'count(foobar)', width: 123})).toEqual({
  181. kind: 'function',
  182. function: ['count', 'foobar', undefined, undefined],
  183. });
  184. // custom tag
  185. expect(explodeField({field: 'foo.bar.is-Enterprise_42', width: 123})).toEqual({
  186. kind: 'field',
  187. field: 'foo.bar.is-Enterprise_42',
  188. });
  189. // custom tag with aggregation
  190. expect(explodeField({field: 'count(foo.bar.is-Enterprise_42)', width: 123})).toEqual({
  191. kind: 'function',
  192. function: ['count', 'foo.bar.is-Enterprise_42', undefined, undefined],
  193. });
  194. });
  195. });
  196. describe('aggregateOutputType', function () {
  197. it('handles unknown fields', function () {
  198. expect(aggregateOutputType('')).toEqual('number');
  199. expect(aggregateOutputType('blerg')).toEqual('number');
  200. });
  201. it('handles duration functions', function () {
  202. expect(aggregateOutputType('p50()')).toEqual('duration');
  203. expect(aggregateOutputType('p75()')).toEqual('duration');
  204. expect(aggregateOutputType('p95()')).toEqual('duration');
  205. expect(aggregateOutputType('p99()')).toEqual('duration');
  206. expect(aggregateOutputType('p100()')).toEqual('duration');
  207. expect(aggregateOutputType('p50(transaction.duration)')).toEqual('duration');
  208. expect(aggregateOutputType('p75(transaction.duration)')).toEqual('duration');
  209. expect(aggregateOutputType('p95(transaction.duration)')).toEqual('duration');
  210. expect(aggregateOutputType('p99(transaction.duration)')).toEqual('duration');
  211. expect(aggregateOutputType('p100(transaction.duration)')).toEqual('duration');
  212. expect(aggregateOutputType('percentile(transaction.duration, 0.51)')).toEqual(
  213. 'duration'
  214. );
  215. expect(aggregateOutputType('percentile(transaction.duration,0.99)')).toEqual(
  216. 'duration'
  217. );
  218. });
  219. it('handles percentage functions', function () {
  220. expect(aggregateOutputType('failure_rate()')).toEqual('percentage');
  221. });
  222. it('handles number functions', function () {
  223. expect(aggregateOutputType('apdex()')).toEqual('number');
  224. expect(aggregateOutputType('apdex(500)')).toEqual('number');
  225. expect(aggregateOutputType('count_miserable(user, 500)')).toEqual('number');
  226. expect(aggregateOutputType('user_misery(500)')).toEqual('number');
  227. expect(aggregateOutputType('eps()')).toEqual('number');
  228. expect(aggregateOutputType('epm()')).toEqual('number');
  229. });
  230. it('handles inherit functions', function () {
  231. expect(aggregateOutputType('sum(transaction.duration)')).toEqual('duration');
  232. expect(aggregateOutputType('sum(stack.colno)')).toEqual('number');
  233. expect(aggregateOutputType('min(stack.colno)')).toEqual('number');
  234. expect(aggregateOutputType('min(timestamp)')).toEqual('date');
  235. expect(aggregateOutputType('max(stack.colno)')).toEqual('number');
  236. expect(aggregateOutputType('max(timestamp)')).toEqual('date');
  237. });
  238. it('handles measurements', function () {
  239. expect(aggregateOutputType('sum(measurements.fcp)')).toEqual('duration');
  240. expect(aggregateOutputType('min(measurements.fcp)')).toEqual('duration');
  241. expect(aggregateOutputType('max(measurements.fcp)')).toEqual('duration');
  242. expect(aggregateOutputType('avg(measurements.fcp)')).toEqual('duration');
  243. expect(aggregateOutputType('percentile(measurements.fcp, 0.5)')).toEqual('duration');
  244. expect(aggregateOutputType('sum(measurements.bar)')).toEqual('number');
  245. expect(aggregateOutputType('min(measurements.bar)')).toEqual('number');
  246. expect(aggregateOutputType('max(measurements.bar)')).toEqual('number');
  247. expect(aggregateOutputType('avg(measurements.bar)')).toEqual('number');
  248. expect(aggregateOutputType('percentile(measurements.bar, 0.5)')).toEqual('number');
  249. expect(aggregateOutputType('p50(measurements.bar)')).toEqual('number');
  250. expect(aggregateOutputType('p75(measurements.bar)')).toEqual('number');
  251. expect(aggregateOutputType('p95(measurements.bar)')).toEqual('number');
  252. expect(aggregateOutputType('p99(measurements.bar)')).toEqual('number');
  253. expect(aggregateOutputType('p100(measurements.bar)')).toEqual('number');
  254. });
  255. });
  256. describe('aggregateMultiPlotType', function () {
  257. it('handles unknown functions', function () {
  258. expect(aggregateMultiPlotType('blerg')).toBe('area');
  259. expect(aggregateMultiPlotType('blerg(uhoh)')).toBe('area');
  260. });
  261. it('handles known functions', function () {
  262. expect(aggregateMultiPlotType('sum(transaction.duration)')).toBe('area');
  263. expect(aggregateMultiPlotType('p95()')).toBe('line');
  264. expect(aggregateMultiPlotType('equation|sum(transaction.duration) / 2')).toBe('line');
  265. });
  266. });
  267. describe('generateAggregateFields', function () {
  268. const organization = Organization();
  269. it('gets default aggregates', function () {
  270. expect(generateAggregateFields(organization, [])).toContainEqual({field: 'count()'});
  271. });
  272. it('includes fields from eventFields', function () {
  273. expect(
  274. generateAggregateFields(organization, [{field: 'not_real_aggregate()'}])
  275. ).toContainEqual({field: 'not_real_aggregate()'});
  276. });
  277. it('excludes fields from aggregates', function () {
  278. expect(generateAggregateFields(organization, [], ['count()'])).not.toContainEqual({
  279. field: 'count()',
  280. });
  281. });
  282. });
  283. describe('fieldAlignment()', function () {
  284. it('works with only field name', function () {
  285. expect(fieldAlignment('event.type')).toEqual('left');
  286. // Should be right, but we don't have any type data.
  287. expect(fieldAlignment('transaction.duration')).toEqual('left');
  288. });
  289. it('works with type parameter', function () {
  290. expect(fieldAlignment('transaction.duration', 'duration')).toEqual('right');
  291. expect(fieldAlignment('device.battery_level', 'number')).toEqual('right');
  292. expect(fieldAlignment('min(timestamp)', 'datetime')).toEqual('left');
  293. });
  294. it('can use table metadata', function () {
  295. const meta = {
  296. 'transaction.duration': 'duration',
  297. title: 'string',
  298. };
  299. expect(fieldAlignment('transaction.duration', 'never', meta)).toEqual('right');
  300. expect(fieldAlignment('transaction.duration', undefined, meta)).toEqual('right');
  301. expect(fieldAlignment('title', undefined, meta)).toEqual('left');
  302. });
  303. });