index.spec.jsx 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384
  1. import React from 'react';
  2. import {initializeOrg} from 'sentry-test/initializeOrg';
  3. import {mockRouterPush} from 'sentry-test/mockRouterPush';
  4. import {mountWithTheme} from 'sentry-test/enzyme';
  5. import {setActiveOrganization} from 'app/actionCreators/organizations';
  6. import GlobalSelectionStore from 'app/stores/globalSelectionStore';
  7. import EventsContainer from 'app/views/events';
  8. import ProjectsStore from 'app/stores/projectsStore';
  9. describe('EventsContainer', function() {
  10. let wrapper;
  11. const environments = ['production', 'staging'];
  12. const params = {
  13. orgId: 'org-slug',
  14. };
  15. const {organization, router, routerContext} = initializeOrg({
  16. projects: [
  17. {isMember: true, environments, isBookmarked: true},
  18. {isMember: true, slug: 'new-project', id: 3, environments},
  19. ],
  20. organization: {
  21. features: ['events', 'global-views'],
  22. },
  23. router: {
  24. location: {
  25. pathname: '/organizations/org-slug/events/',
  26. query: {},
  27. },
  28. params,
  29. },
  30. });
  31. beforeAll(async function() {
  32. MockApiClient.addMockResponse({
  33. url: '/organizations/org-slug/projects/',
  34. body: [],
  35. });
  36. MockApiClient.addMockResponse({
  37. url: '/organizations/org-slug/tags/',
  38. body: [
  39. {count: 1, key: 'transaction', name: 'Transaction'},
  40. {count: 2, key: 'mechanism', name: 'Mechanism'},
  41. ],
  42. });
  43. setActiveOrganization(organization);
  44. await tick();
  45. });
  46. describe('Header', function() {
  47. beforeEach(function() {
  48. GlobalSelectionStore.reset();
  49. ProjectsStore.loadInitialData(organization.projects);
  50. router.location = {
  51. pathname: '/organizations/org-slug/events/',
  52. query: {},
  53. };
  54. wrapper = mountWithTheme(
  55. <EventsContainer
  56. router={router}
  57. organization={organization}
  58. location={router.location}
  59. >
  60. <div />
  61. </EventsContainer>,
  62. routerContext
  63. );
  64. mockRouterPush(wrapper, router);
  65. });
  66. it('updates router when changing environments', async function() {
  67. expect(wrapper.find('PageContent')).toHaveLength(1);
  68. expect(wrapper.find('MultipleEnvironmentSelector').prop('value')).toEqual([]);
  69. wrapper.find('MultipleEnvironmentSelector HeaderItem').simulate('click');
  70. await tick();
  71. wrapper.update();
  72. wrapper
  73. .find('EnvironmentSelectorItem')
  74. .at(0)
  75. .simulate('click');
  76. await tick();
  77. wrapper.update();
  78. expect(router.push).toHaveBeenLastCalledWith({
  79. pathname: '/organizations/org-slug/events/',
  80. query: {
  81. environment: ['production'],
  82. },
  83. });
  84. await tick();
  85. wrapper.update();
  86. expect(wrapper.find('MultipleEnvironmentSelector').prop('value')).toEqual([
  87. 'production',
  88. ]);
  89. // Select a second environment, "staging"
  90. wrapper.find('MultipleEnvironmentSelector HeaderItem').simulate('click');
  91. await tick();
  92. wrapper.update();
  93. wrapper
  94. .find('EnvironmentSelectorItem')
  95. .at(1)
  96. .find('CheckboxHitbox')
  97. .simulate('click');
  98. expect(wrapper.find('MultipleEnvironmentSelector').prop('value')).toEqual([
  99. 'production',
  100. 'staging',
  101. ]);
  102. // close dropdown
  103. wrapper
  104. .find('MultipleEnvironmentSelector StyledInput')
  105. .simulate('keyDown', {key: 'Escape'});
  106. await tick();
  107. wrapper.update();
  108. expect(router.push).toHaveBeenLastCalledWith({
  109. pathname: '/organizations/org-slug/events/',
  110. query: {
  111. environment: ['production', 'staging'],
  112. },
  113. });
  114. expect(wrapper.find('MultipleEnvironmentSelector').prop('value')).toEqual([
  115. 'production',
  116. 'staging',
  117. ]);
  118. // Can clear
  119. wrapper.find('MultipleEnvironmentSelector HeaderItem').simulate('click');
  120. await tick();
  121. wrapper.update();
  122. wrapper
  123. .find('MultipleEnvironmentSelector HeaderItem StyledClose')
  124. .simulate('click');
  125. await tick();
  126. wrapper.update();
  127. expect(wrapper.find('MultipleEnvironmentSelector').prop('value')).toEqual([]);
  128. expect(router.push).toHaveBeenCalledWith({
  129. pathname: '/organizations/org-slug/events/',
  130. query: {
  131. environment: [],
  132. },
  133. });
  134. });
  135. it('updates router when changing projects', async function() {
  136. expect(wrapper.find('MultipleProjectSelector').prop('value')).toEqual([]);
  137. wrapper.find('MultipleProjectSelector HeaderItem').simulate('click');
  138. wrapper
  139. .find('MultipleProjectSelector AutoCompleteItem ProjectSelectorItem')
  140. .first()
  141. .simulate('click');
  142. await tick();
  143. wrapper.update();
  144. expect(router.push).toHaveBeenCalledWith({
  145. pathname: '/organizations/org-slug/events/',
  146. query: {
  147. project: [2],
  148. },
  149. });
  150. expect(wrapper.find('MultipleProjectSelector').prop('value')).toEqual([2]);
  151. });
  152. it('selects multiple projects', async function() {
  153. expect(wrapper.find('MultipleProjectSelector').prop('value')).toEqual([]);
  154. wrapper.find('MultipleProjectSelector HeaderItem').simulate('click');
  155. wrapper
  156. .find('MultipleProjectSelector AutoCompleteItem CheckboxHitbox')
  157. .at(0)
  158. .simulate('click');
  159. expect(wrapper.find('MultipleProjectSelector').prop('value')).toEqual([2]);
  160. wrapper
  161. .find('MultipleProjectSelector AutoCompleteItem CheckboxHitbox')
  162. .at(1)
  163. .simulate('click');
  164. expect(wrapper.find('MultipleProjectSelector').prop('value')).toEqual([2, 3]);
  165. wrapper.find('MultipleProjectSelector StyledChevron').simulate('click');
  166. expect(router.push).toHaveBeenCalledWith({
  167. pathname: '/organizations/org-slug/events/',
  168. query: {
  169. project: [2, 3],
  170. },
  171. });
  172. });
  173. it('changes to absolute time (utc is default)', async function() {
  174. const start = new Date('2017-10-01T00:00:00.000Z');
  175. const end = new Date('2017-10-01T23:59:59.000Z');
  176. wrapper.find('TimeRangeSelector HeaderItem').simulate('click');
  177. await wrapper.find('SelectorItem[value="absolute"]').simulate('click');
  178. // Oct 1st
  179. wrapper
  180. .find('DayCell')
  181. .at(0)
  182. .simulate('mouseUp');
  183. expect(wrapper.find('UtcPicker Checkbox').prop('checked')).toBe(true);
  184. wrapper.find('TimeRangeSelector StyledChevron').simulate('click');
  185. await tick();
  186. wrapper.update();
  187. expect(router.push).toHaveBeenCalledWith({
  188. pathname: '/organizations/org-slug/events/',
  189. query: {
  190. start: '2017-10-01T00:00:00',
  191. end: '2017-10-01T23:59:59',
  192. utc: true,
  193. },
  194. });
  195. expect(wrapper.find('TimeRangeSelector').prop('start')).toEqual(start);
  196. expect(wrapper.find('TimeRangeSelector').prop('end')).toEqual(end);
  197. expect(wrapper.find('TimeRangeSelector').prop('relative')).toEqual(null);
  198. // Open menu and make sure UTC is checked
  199. wrapper.find('TimeRangeSelector HeaderItem').simulate('click');
  200. await tick();
  201. wrapper.update();
  202. expect(wrapper.find('UtcPicker Checkbox').prop('checked')).toBe(true);
  203. });
  204. it('does not update router when toggling environment selector without changes', async function() {
  205. const prevCallCount = router.push.mock.calls.length;
  206. wrapper.setProps({
  207. router: {
  208. ...router,
  209. location: {
  210. ...router.location,
  211. query: {
  212. environment: ['production'],
  213. utc: 'true',
  214. },
  215. },
  216. },
  217. });
  218. // Toggle MultipleProjectSelector
  219. wrapper.find('MultipleEnvironmentSelector HeaderItem').simulate('click');
  220. wrapper
  221. .find('MultipleEnvironmentSelector StyledInput')
  222. .simulate('keyDown', {key: 'Escape'});
  223. expect(router.push).toHaveBeenCalledTimes(prevCallCount);
  224. });
  225. it('updates router when changing periods', async function() {
  226. expect(wrapper.find('TimeRangeSelector').prop('start')).toEqual(null);
  227. expect(wrapper.find('TimeRangeSelector').prop('end')).toEqual(null);
  228. expect(wrapper.find('TimeRangeSelector').prop('relative')).toEqual('14d');
  229. wrapper.find('TimeRangeSelector HeaderItem').simulate('click');
  230. expect(wrapper.find('[data-test-id="date-range"]')).toHaveLength(0);
  231. wrapper.find('SelectorItem[value="absolute"]').simulate('click');
  232. wrapper.find('TimeRangeSelector HeaderItem').simulate('click');
  233. await tick();
  234. wrapper.update();
  235. // The current date in local timezone is 2017-10-16T22:41:20-04:00
  236. // we take the local date and subtract 14d
  237. // If "UTC" override is true then we strip timezone and use that date for the range
  238. expect(router.push).toHaveBeenCalledWith({
  239. pathname: '/organizations/org-slug/events/',
  240. query: {
  241. end: '2017-10-16T22:41:20',
  242. start: '2017-10-02T22:41:20',
  243. utc: true,
  244. },
  245. });
  246. expect(wrapper.find('TimeRangeSelector').props()).toEqual(
  247. expect.objectContaining({
  248. end: new Date('2017-10-16T22:41:20.000Z'),
  249. start: new Date('2017-10-02T22:41:20.000Z'),
  250. utc: true,
  251. })
  252. );
  253. // Can switch back to relative date
  254. wrapper.find('TimeRangeSelector HeaderItem').simulate('click');
  255. wrapper.find('SelectorItem[value="7d"]').simulate('click');
  256. wrapper.find('TimeRangeSelector HeaderItem').simulate('click');
  257. await tick();
  258. wrapper.update();
  259. expect(wrapper.find('TimeRangeSelector').prop('relative')).toEqual('7d');
  260. // Note this is NOT called with utc true like the above because 1) it's a relative date
  261. // and we do not need UTC value (which is default true in tests because timezone is set to UTC)
  262. // and 2) we removed forcing a default value in url params so there is no explicit utc value
  263. expect(router.push).toHaveBeenCalledWith({
  264. pathname: '/organizations/org-slug/events/',
  265. query: {
  266. statsPeriod: '7d',
  267. },
  268. });
  269. expect(wrapper.find('TimeRangeSelector').props()).toEqual(
  270. expect.objectContaining({
  271. end: null,
  272. start: null,
  273. relative: '7d',
  274. })
  275. );
  276. });
  277. it('updates TimeRangeSelector when changing routes', async function() {
  278. let newRouter = {
  279. router: {
  280. ...router,
  281. location: {
  282. pathname: '/organizations/org-slug/events2/',
  283. query: {
  284. end: '2017-10-17T02:41:20',
  285. start: '2017-10-03T02:41:20',
  286. utc: 'true',
  287. },
  288. },
  289. },
  290. };
  291. wrapper.setProps(newRouter);
  292. wrapper.setContext(newRouter);
  293. await tick();
  294. wrapper.update();
  295. expect(wrapper.find('TimeRangeSelector').text()).toEqual(
  296. 'Oct 3, 201702:41toOct 17, 201702:41'
  297. );
  298. newRouter = {
  299. router: {
  300. ...router,
  301. location: {
  302. pathname: '/organizations/org-slug/events/',
  303. query: {
  304. statsPeriod: '7d',
  305. end: null,
  306. start: null,
  307. utc: 'true',
  308. },
  309. },
  310. },
  311. };
  312. wrapper.setProps(newRouter);
  313. wrapper.setContext(newRouter);
  314. await tick();
  315. wrapper.update();
  316. expect(wrapper.find('TimeRangeSelector').text()).toEqual('Last 7 days');
  317. });
  318. });
  319. });