cellAction.spec.jsx 15 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467
  1. import {render, screen, userEvent} from 'sentry-test/reactTestingLibrary';
  2. import EventView from 'sentry/utils/discover/eventView';
  3. import {MutableSearch} from 'sentry/utils/tokenizeSearch';
  4. import CellAction, {Actions, updateQuery} from 'sentry/views/discover/table/cellAction';
  5. const defaultData = {
  6. transaction: 'best-transaction',
  7. count: 19,
  8. timestamp: '2020-06-09T01:46:25+00:00',
  9. release: 'F2520C43515BD1F0E8A6BD46233324641A370BF6',
  10. nullValue: null,
  11. 'measurements.fcp': 1234,
  12. 'percentile(measurements.fcp, 0.5)': 1234,
  13. 'error.handled': [null],
  14. 'error.type': [
  15. 'ServerException',
  16. 'ClickhouseError',
  17. 'QueryException',
  18. 'QueryException',
  19. ],
  20. };
  21. function renderComponent(
  22. eventView,
  23. handleCellAction,
  24. columnIndex = 0,
  25. data = defaultData
  26. ) {
  27. return render(
  28. <CellAction
  29. dataRow={data}
  30. eventView={eventView}
  31. column={eventView.getColumns()[columnIndex]}
  32. handleCellAction={handleCellAction}
  33. >
  34. <strong>some content</strong>
  35. </CellAction>
  36. );
  37. }
  38. describe('Discover -> CellAction', function () {
  39. const location = {
  40. query: {
  41. id: '42',
  42. name: 'best query',
  43. field: [
  44. 'transaction',
  45. 'count()',
  46. 'timestamp',
  47. 'release',
  48. 'nullValue',
  49. 'measurements.fcp',
  50. 'percentile(measurements.fcp, 0.5)',
  51. 'error.handled',
  52. 'error.type',
  53. ],
  54. widths: ['437', '647', '416', '905'],
  55. sort: ['title'],
  56. query: 'event.type:transaction',
  57. project: [123],
  58. start: '2019-10-01T00:00:00',
  59. end: '2019-10-02T00:00:00',
  60. statsPeriod: '14d',
  61. environment: ['staging'],
  62. yAxis: 'p95',
  63. },
  64. };
  65. const view = EventView.fromLocation(location);
  66. async function hoverContainer() {
  67. await userEvent.hover(screen.getByText('some content'));
  68. }
  69. async function unhoverContainer() {
  70. await userEvent.unhover(screen.getByText('some content'));
  71. }
  72. async function openMenu() {
  73. await hoverContainer();
  74. await userEvent.click(screen.getByRole('button'));
  75. }
  76. describe('hover menu button', function () {
  77. it('shows no menu by default', function () {
  78. renderComponent(view);
  79. expect(screen.queryByRole('button')).not.toBeInTheDocument();
  80. });
  81. it('shows a menu on hover, and hides again', async function () {
  82. renderComponent(view);
  83. await hoverContainer();
  84. expect(screen.getByRole('button')).toBeInTheDocument();
  85. await unhoverContainer();
  86. expect(screen.queryByRole('button')).not.toBeInTheDocument();
  87. });
  88. });
  89. describe('opening the menu', function () {
  90. it('toggles the menu on click', async function () {
  91. renderComponent(view);
  92. await openMenu();
  93. expect(screen.getByRole('button', {name: 'Add to filter'})).toBeInTheDocument();
  94. });
  95. });
  96. describe('per cell actions', function () {
  97. let handleCellAction;
  98. beforeEach(function () {
  99. handleCellAction = jest.fn();
  100. });
  101. it('add button appends condition', async function () {
  102. renderComponent(view, handleCellAction);
  103. await openMenu();
  104. await userEvent.click(screen.getByRole('button', {name: 'Add to filter'}));
  105. expect(handleCellAction).toHaveBeenCalledWith('add', 'best-transaction');
  106. });
  107. it('exclude button adds condition', async function () {
  108. renderComponent(view, handleCellAction);
  109. await openMenu();
  110. await userEvent.click(screen.getByRole('button', {name: 'Exclude from filter'}));
  111. expect(handleCellAction).toHaveBeenCalledWith('exclude', 'best-transaction');
  112. });
  113. it('exclude button appends exclusions', async function () {
  114. const excludeView = EventView.fromLocation({
  115. query: {...location.query, query: '!transaction:nope'},
  116. });
  117. renderComponent(excludeView, handleCellAction);
  118. await openMenu();
  119. await userEvent.click(screen.getByRole('button', {name: 'Exclude from filter'}));
  120. expect(handleCellAction).toHaveBeenCalledWith('exclude', 'best-transaction');
  121. });
  122. it('go to release button goes to release health page', async function () {
  123. renderComponent(view, handleCellAction, 3);
  124. await openMenu();
  125. await userEvent.click(screen.getByRole('button', {name: 'Go to release'}));
  126. expect(handleCellAction).toHaveBeenCalledWith(
  127. 'release',
  128. 'F2520C43515BD1F0E8A6BD46233324641A370BF6'
  129. );
  130. });
  131. it('greater than button adds condition', async function () {
  132. renderComponent(view, handleCellAction, 2);
  133. await openMenu();
  134. await userEvent.click(
  135. screen.getByRole('button', {name: 'Show values greater than'})
  136. );
  137. expect(handleCellAction).toHaveBeenCalledWith(
  138. 'show_greater_than',
  139. '2020-06-09T01:46:25+00:00'
  140. );
  141. });
  142. it('less than button adds condition', async function () {
  143. renderComponent(view, handleCellAction, 2);
  144. await openMenu();
  145. await userEvent.click(screen.getByRole('button', {name: 'Show values less than'}));
  146. expect(handleCellAction).toHaveBeenCalledWith(
  147. 'show_less_than',
  148. '2020-06-09T01:46:25+00:00'
  149. );
  150. });
  151. it('error.handled with null adds condition', async function () {
  152. renderComponent(view, handleCellAction, 7, defaultData);
  153. await openMenu();
  154. await userEvent.click(screen.getByRole('button', {name: 'Add to filter'}));
  155. expect(handleCellAction).toHaveBeenCalledWith('add', 1);
  156. });
  157. it('error.type with array values adds condition', async function () {
  158. renderComponent(view, handleCellAction, 8, defaultData);
  159. await openMenu();
  160. await userEvent.click(screen.getByRole('button', {name: 'Add to filter'}));
  161. expect(handleCellAction).toHaveBeenCalledWith('add', [
  162. 'ServerException',
  163. 'ClickhouseError',
  164. 'QueryException',
  165. 'QueryException',
  166. ]);
  167. });
  168. it('error.handled with 0 adds condition', async function () {
  169. renderComponent(view, handleCellAction, 7, {
  170. ...defaultData,
  171. 'error.handled': [0],
  172. });
  173. await openMenu();
  174. await userEvent.click(screen.getByRole('button', {name: 'Add to filter'}));
  175. expect(handleCellAction).toHaveBeenCalledWith('add', [0]);
  176. });
  177. it('show appropriate actions for string cells', async function () {
  178. renderComponent(view, handleCellAction, 0);
  179. await openMenu();
  180. expect(screen.getByRole('button', {name: 'Add to filter'})).toBeInTheDocument();
  181. expect(
  182. screen.getByRole('button', {name: 'Exclude from filter'})
  183. ).toBeInTheDocument();
  184. expect(
  185. screen.queryByRole('button', {name: 'Show values greater than'})
  186. ).not.toBeInTheDocument();
  187. expect(
  188. screen.queryByRole('button', {name: 'Show values less than'})
  189. ).not.toBeInTheDocument();
  190. });
  191. it('show appropriate actions for string cells with null values', async function () {
  192. renderComponent(view, handleCellAction, 4);
  193. await openMenu();
  194. expect(screen.getByRole('button', {name: 'Add to filter'})).toBeInTheDocument();
  195. expect(
  196. screen.getByRole('button', {name: 'Exclude from filter'})
  197. ).toBeInTheDocument();
  198. });
  199. it('show appropriate actions for number cells', async function () {
  200. renderComponent(view, handleCellAction, 1);
  201. await openMenu();
  202. expect(
  203. screen.queryByRole('button', {name: 'Add to filter'})
  204. ).not.toBeInTheDocument();
  205. expect(
  206. screen.queryByRole('button', {name: 'Exclude from filter'})
  207. ).not.toBeInTheDocument();
  208. expect(
  209. screen.getByRole('button', {name: 'Show values greater than'})
  210. ).toBeInTheDocument();
  211. expect(
  212. screen.getByRole('button', {name: 'Show values less than'})
  213. ).toBeInTheDocument();
  214. });
  215. it('show appropriate actions for date cells', async function () {
  216. renderComponent(view, handleCellAction, 2);
  217. await openMenu();
  218. expect(screen.getByRole('button', {name: 'Add to filter'})).toBeInTheDocument();
  219. expect(
  220. screen.queryByRole('button', {name: 'Exclude from filter'})
  221. ).not.toBeInTheDocument();
  222. expect(
  223. screen.getByRole('button', {name: 'Show values greater than'})
  224. ).toBeInTheDocument();
  225. expect(
  226. screen.getByRole('button', {name: 'Show values less than'})
  227. ).toBeInTheDocument();
  228. });
  229. it('show appropriate actions for release cells', async function () {
  230. renderComponent(view, handleCellAction, 3);
  231. await openMenu();
  232. expect(screen.getByRole('button', {name: 'Go to release'})).toBeInTheDocument();
  233. });
  234. it('show appropriate actions for empty release cells', async function () {
  235. renderComponent(view, handleCellAction, 3, {...defaultData, release: null});
  236. await openMenu();
  237. expect(
  238. screen.queryByRole('button', {name: 'Go to release'})
  239. ).not.toBeInTheDocument();
  240. });
  241. it('show appropriate actions for measurement cells', async function () {
  242. renderComponent(view, handleCellAction, 5);
  243. await openMenu();
  244. expect(
  245. screen.queryByRole('button', {name: 'Add to filter'})
  246. ).not.toBeInTheDocument();
  247. expect(
  248. screen.queryByRole('button', {name: 'Exclude from filter'})
  249. ).not.toBeInTheDocument();
  250. expect(
  251. screen.getByRole('button', {name: 'Show values greater than'})
  252. ).toBeInTheDocument();
  253. expect(
  254. screen.getByRole('button', {name: 'Show values less than'})
  255. ).toBeInTheDocument();
  256. });
  257. it('show appropriate actions for empty measurement cells', async function () {
  258. renderComponent(view, handleCellAction, 5, {
  259. ...defaultData,
  260. 'measurements.fcp': null,
  261. });
  262. await openMenu();
  263. expect(screen.getByRole('button', {name: 'Add to filter'})).toBeInTheDocument();
  264. expect(
  265. screen.getByRole('button', {name: 'Exclude from filter'})
  266. ).toBeInTheDocument();
  267. expect(
  268. screen.queryByRole('button', {name: 'Show values greater than'})
  269. ).not.toBeInTheDocument();
  270. expect(
  271. screen.queryByRole('button', {name: 'Show values less than'})
  272. ).not.toBeInTheDocument();
  273. });
  274. it('show appropriate actions for numeric function cells', async function () {
  275. renderComponent(view, handleCellAction, 6);
  276. await openMenu();
  277. expect(
  278. screen.getByRole('button', {name: 'Show values greater than'})
  279. ).toBeInTheDocument();
  280. expect(
  281. screen.getByRole('button', {name: 'Show values less than'})
  282. ).toBeInTheDocument();
  283. });
  284. it('show appropriate actions for empty numeric function cells', async function () {
  285. renderComponent(view, handleCellAction, 6, {
  286. ...defaultData,
  287. 'percentile(measurements.fcp, 0.5)': null,
  288. });
  289. await hoverContainer();
  290. expect(screen.queryByRole('button')).not.toBeInTheDocument();
  291. });
  292. });
  293. });
  294. describe('updateQuery()', function () {
  295. const columnA = {
  296. key: 'a',
  297. name: 'a',
  298. type: 'number',
  299. isSortable: false,
  300. column: {
  301. kind: 'field',
  302. field: 'a',
  303. },
  304. width: -1,
  305. };
  306. const columnB = {
  307. key: 'b',
  308. name: 'b',
  309. type: 'number',
  310. isSortable: false,
  311. column: {
  312. kind: 'field',
  313. field: 'b',
  314. },
  315. width: -1,
  316. };
  317. it('modifies the query with has/!has', function () {
  318. let results = new MutableSearch([]);
  319. updateQuery(results, Actions.ADD, columnA, null);
  320. expect(results.formatString()).toEqual('!has:a');
  321. updateQuery(results, Actions.EXCLUDE, columnA, null);
  322. expect(results.formatString()).toEqual('has:a');
  323. updateQuery(results, Actions.ADD, columnA, null);
  324. expect(results.formatString()).toEqual('!has:a');
  325. results = new MutableSearch([]);
  326. updateQuery(results, Actions.ADD, columnA, [null]);
  327. expect(results.formatString()).toEqual('!has:a');
  328. });
  329. it('modifies the query with additions', function () {
  330. const results = new MutableSearch([]);
  331. updateQuery(results, Actions.ADD, columnA, '1');
  332. expect(results.formatString()).toEqual('a:1');
  333. updateQuery(results, Actions.ADD, columnB, '1');
  334. expect(results.formatString()).toEqual('a:1 b:1');
  335. updateQuery(results, Actions.ADD, columnA, '2');
  336. expect(results.formatString()).toEqual('b:1 a:2');
  337. updateQuery(results, Actions.ADD, columnA, ['1', '2', '3']);
  338. expect(results.formatString()).toEqual('b:1 a:2 a:1 a:3');
  339. });
  340. it('modifies the query with exclusions', function () {
  341. const results = new MutableSearch([]);
  342. updateQuery(results, Actions.EXCLUDE, columnA, '1');
  343. expect(results.formatString()).toEqual('!a:1');
  344. updateQuery(results, Actions.EXCLUDE, columnB, '1');
  345. expect(results.formatString()).toEqual('!a:1 !b:1');
  346. updateQuery(results, Actions.EXCLUDE, columnA, '2');
  347. expect(results.formatString()).toEqual('!b:1 !a:1 !a:2');
  348. updateQuery(results, Actions.EXCLUDE, columnA, ['1', '2', '3']);
  349. expect(results.formatString()).toEqual('!b:1 !a:1 !a:2 !a:3');
  350. });
  351. it('modifies the query with a mix of additions and exclusions', function () {
  352. const results = new MutableSearch([]);
  353. updateQuery(results, Actions.ADD, columnA, '1');
  354. expect(results.formatString()).toEqual('a:1');
  355. updateQuery(results, Actions.ADD, columnB, '2');
  356. expect(results.formatString()).toEqual('a:1 b:2');
  357. updateQuery(results, Actions.EXCLUDE, columnA, '3');
  358. expect(results.formatString()).toEqual('b:2 !a:3');
  359. updateQuery(results, Actions.EXCLUDE, columnB, '4');
  360. expect(results.formatString()).toEqual('!a:3 !b:4');
  361. results.addFilterValues('!a', ['*dontescapeme*'], false);
  362. expect(results.formatString()).toEqual('!a:3 !b:4 !a:*dontescapeme*');
  363. updateQuery(results, Actions.EXCLUDE, columnA, '*escapeme*');
  364. expect(results.formatString()).toEqual(
  365. '!b:4 !a:3 !a:*dontescapeme* !a:"\\*escapeme\\*"'
  366. );
  367. updateQuery(results, Actions.ADD, columnA, '5');
  368. expect(results.formatString()).toEqual('!b:4 a:5');
  369. updateQuery(results, Actions.ADD, columnB, '6');
  370. expect(results.formatString()).toEqual('a:5 b:6');
  371. });
  372. it('modifies the query with greater/less than', function () {
  373. const results = new MutableSearch([]);
  374. updateQuery(results, Actions.SHOW_GREATER_THAN, columnA, 1);
  375. expect(results.formatString()).toEqual('a:>1');
  376. updateQuery(results, Actions.SHOW_GREATER_THAN, columnA, 2);
  377. expect(results.formatString()).toEqual('a:>2');
  378. updateQuery(results, Actions.SHOW_LESS_THAN, columnA, 3);
  379. expect(results.formatString()).toEqual('a:<3');
  380. updateQuery(results, Actions.SHOW_LESS_THAN, columnA, 4);
  381. expect(results.formatString()).toEqual('a:<4');
  382. });
  383. it('modifies the query with greater/less than on duration fields', function () {
  384. const columnADuration = {...columnA, type: 'duration'};
  385. const results = new MutableSearch([]);
  386. updateQuery(results, Actions.SHOW_GREATER_THAN, columnADuration, 1);
  387. expect(results.formatString()).toEqual('a:>1.00ms');
  388. updateQuery(results, Actions.SHOW_GREATER_THAN, columnADuration, 2);
  389. expect(results.formatString()).toEqual('a:>2.00ms');
  390. updateQuery(results, Actions.SHOW_LESS_THAN, columnADuration, 3);
  391. expect(results.formatString()).toEqual('a:<3.00ms');
  392. updateQuery(results, Actions.SHOW_LESS_THAN, columnADuration, 4.1234);
  393. expect(results.formatString()).toEqual('a:<4.12ms');
  394. });
  395. it('does not error for special actions', function () {
  396. const results = new MutableSearch([]);
  397. updateQuery(results, Actions.RELEASE, columnA, '');
  398. updateQuery(results, Actions.DRILLDOWN, columnA, '');
  399. });
  400. it('errors for unknown actions', function () {
  401. const results = new MutableSearch([]);
  402. expect(() => updateQuery(results, 'unknown', columnA, '')).toThrow();
  403. });
  404. });