assigneeSelector.spec.jsx 10 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336
  1. import React from 'react';
  2. import {
  3. AssigneeSelectorComponent,
  4. putSessionUserFirst,
  5. } from 'app/components/assigneeSelector';
  6. import {Client} from 'app/api';
  7. import {mountWithTheme} from 'sentry-test/enzyme';
  8. import ConfigStore from 'app/stores/configStore';
  9. import GroupStore from 'app/stores/groupStore';
  10. import MemberListStore from 'app/stores/memberListStore';
  11. import ProjectsStore from 'app/stores/projectsStore';
  12. import TeamStore from 'app/stores/teamStore';
  13. describe('AssigneeSelector', function() {
  14. let assigneeSelector;
  15. let assignMock;
  16. let openMenu;
  17. let USER_1, USER_2, USER_3;
  18. let TEAM_1;
  19. let PROJECT_1;
  20. let GROUP_1;
  21. beforeEach(function() {
  22. USER_1 = TestStubs.User({
  23. id: '1',
  24. name: 'Jane Doe',
  25. email: 'janedoe@example.com',
  26. });
  27. USER_2 = TestStubs.User({
  28. id: '2',
  29. name: 'John Smith',
  30. email: 'johnsmith@example.com',
  31. });
  32. USER_3 = TestStubs.User({
  33. id: '3',
  34. name: 'J J',
  35. email: 'jj@example.com',
  36. });
  37. TEAM_1 = TestStubs.Team({
  38. id: '3',
  39. name: 'COOL TEAM',
  40. slug: 'cool-team',
  41. });
  42. PROJECT_1 = TestStubs.Project({
  43. teams: [TEAM_1],
  44. });
  45. GROUP_1 = TestStubs.Group({
  46. id: '1337',
  47. project: {
  48. id: PROJECT_1.id,
  49. slug: PROJECT_1.slug,
  50. },
  51. });
  52. jest.spyOn(MemberListStore, 'getAll').mockImplementation(() => null);
  53. jest.spyOn(TeamStore, 'getAll').mockImplementation(() => [TEAM_1]);
  54. jest.spyOn(ProjectsStore, 'getAll').mockImplementation(() => [PROJECT_1]);
  55. jest.spyOn(GroupStore, 'get').mockImplementation(() => GROUP_1);
  56. assignMock = Client.addMockResponse({
  57. method: 'PUT',
  58. url: `/issues/${GROUP_1.id}/`,
  59. body: {
  60. ...GROUP_1,
  61. assignedTo: USER_1,
  62. },
  63. });
  64. MemberListStore.items = null;
  65. MemberListStore.loaded = false;
  66. assigneeSelector = mountWithTheme(
  67. <AssigneeSelectorComponent id={GROUP_1.id} />,
  68. TestStubs.routerContext()
  69. );
  70. openMenu = () => assigneeSelector.find('DropdownButton').simulate('click');
  71. });
  72. afterEach(function() {
  73. Client.clearMockResponses();
  74. });
  75. describe('render with props', function() {
  76. it('renders members from the prop when present', async function() {
  77. assigneeSelector = mountWithTheme(
  78. <AssigneeSelectorComponent id={GROUP_1.id} memberList={[USER_2, USER_3]} />,
  79. TestStubs.routerContext()
  80. );
  81. MemberListStore.loadInitialData([USER_1]);
  82. openMenu();
  83. assigneeSelector.update();
  84. expect(assigneeSelector.find('LoadingIndicator')).toHaveLength(0);
  85. expect(assigneeSelector.find('Avatar')).toHaveLength(3);
  86. expect(assigneeSelector.find('UserAvatar')).toHaveLength(2);
  87. expect(assigneeSelector.find('TeamAvatar')).toHaveLength(1);
  88. const names = assigneeSelector
  89. .find('MenuItemWrapper Label Highlight')
  90. .map(el => el.text());
  91. expect(names).toEqual([`#${TEAM_1.slug}`, USER_2.name, USER_3.name]);
  92. });
  93. });
  94. describe('putSessionUserFirst()', function() {
  95. it('should place the session user at the top of the member list if present', function() {
  96. jest.spyOn(ConfigStore, 'get').mockImplementation(() => ({
  97. id: '2',
  98. name: 'John Smith',
  99. email: 'johnsmith@example.com',
  100. }));
  101. expect(putSessionUserFirst([USER_1, USER_2])).toEqual([USER_2, USER_1]);
  102. ConfigStore.get.mockRestore();
  103. });
  104. it("should return the same member list if the session user isn't present", function() {
  105. jest.spyOn(ConfigStore, 'get').mockImplementation(() => ({
  106. id: '555',
  107. name: 'Here Comes a New Challenger',
  108. email: 'guile@mail.us.af.mil',
  109. }));
  110. expect(putSessionUserFirst([USER_1, USER_2])).toEqual([USER_1, USER_2]);
  111. ConfigStore.get.mockRestore();
  112. });
  113. });
  114. it('should initially have loading state', function() {
  115. openMenu();
  116. expect(assigneeSelector.find('LoadingIndicator')).toHaveLength(1);
  117. });
  118. it('does not have loading state and shows member list after calling MemberListStore.loadInitialData', async function() {
  119. openMenu();
  120. MemberListStore.loadInitialData([USER_1, USER_2]);
  121. assigneeSelector.update();
  122. expect(assigneeSelector.instance().assignableTeams()).toHaveLength(1);
  123. expect(assigneeSelector.find('LoadingIndicator')).toHaveLength(0);
  124. expect(assigneeSelector.find('Avatar')).toHaveLength(3);
  125. expect(assigneeSelector.find('UserAvatar')).toHaveLength(2);
  126. expect(assigneeSelector.find('TeamAvatar')).toHaveLength(1);
  127. });
  128. it('does NOT update member list after initial load', function() {
  129. openMenu();
  130. MemberListStore.loadInitialData([USER_1, USER_2]);
  131. assigneeSelector.update();
  132. expect(assigneeSelector.find('Avatar')).toHaveLength(3);
  133. expect(assigneeSelector.find('LoadingIndicator').exists()).toBe(false);
  134. MemberListStore.loadInitialData([USER_1, USER_2, USER_3]);
  135. assigneeSelector.update();
  136. expect(assigneeSelector.find('Avatar')).toHaveLength(3);
  137. expect(assigneeSelector.find('LoadingIndicator').exists()).toBe(false);
  138. });
  139. it('successfully assigns users', async function() {
  140. openMenu();
  141. MemberListStore.loadInitialData([USER_1, USER_2]);
  142. assigneeSelector.update();
  143. expect(assigneeSelector.find('LoadingIndicator').exists()).toBe(false);
  144. assigneeSelector
  145. .find('UserAvatar')
  146. .first()
  147. .simulate('click');
  148. expect(assignMock).toHaveBeenLastCalledWith(
  149. '/issues/1337/',
  150. expect.objectContaining({
  151. data: {assignedTo: 'user:1'},
  152. })
  153. );
  154. assigneeSelector.update();
  155. expect(assigneeSelector.find('LoadingIndicator')).toHaveLength(1);
  156. // Flakey with 1 tick
  157. await tick();
  158. await tick();
  159. assigneeSelector.update();
  160. expect(assigneeSelector.find('LoadingIndicator')).toHaveLength(0);
  161. expect(assigneeSelector.find('ActorAvatar')).toHaveLength(1);
  162. });
  163. it('successfully assigns teams', async function() {
  164. openMenu();
  165. MemberListStore.loadInitialData([USER_1, USER_2]);
  166. assigneeSelector.update();
  167. expect(assigneeSelector.find('LoadingIndicator').exists()).toBe(false);
  168. assigneeSelector
  169. .find('TeamAvatar')
  170. .first()
  171. .simulate('click');
  172. assigneeSelector.update();
  173. expect(assigneeSelector.find('LoadingIndicator').exists()).toBe(true);
  174. expect(assignMock).toHaveBeenCalledWith(
  175. '/issues/1337/',
  176. expect.objectContaining({
  177. data: {assignedTo: 'team:3'},
  178. })
  179. );
  180. // Flakey with 1 tick
  181. await tick();
  182. await tick();
  183. assigneeSelector.update();
  184. expect(assigneeSelector.find('LoadingIndicator').exists()).toBe(false);
  185. expect(assigneeSelector.find('ActorAvatar')).toHaveLength(1);
  186. });
  187. it('successfully clears assignment', async function() {
  188. openMenu();
  189. MemberListStore.loadInitialData([USER_1, USER_2]);
  190. // Assign first item in list, which is TEAM_1
  191. assigneeSelector.update();
  192. assigneeSelector
  193. .find('Avatar')
  194. .first()
  195. .simulate('click');
  196. assigneeSelector.update();
  197. expect(assigneeSelector.find('LoadingIndicator').exists()).toBe(true);
  198. expect(assignMock).toHaveBeenCalledWith(
  199. '/issues/1337/',
  200. expect.objectContaining({
  201. data: {assignedTo: 'team:3'},
  202. })
  203. );
  204. // Waiting for assignment to finish updating
  205. // Flakey with 1 tick
  206. await tick();
  207. await tick();
  208. assigneeSelector.update();
  209. openMenu();
  210. assigneeSelector
  211. .find('MenuItemWrapper[data-test-id="clear-assignee"]')
  212. .simulate('click');
  213. // api was called with empty string, clearing assignment
  214. expect(assignMock).toHaveBeenLastCalledWith(
  215. '/issues/1337/',
  216. expect.objectContaining({
  217. data: {assignedTo: ''},
  218. })
  219. );
  220. });
  221. it('shows invite member button', async function() {
  222. const routerContext = TestStubs.routerContext();
  223. openMenu();
  224. MemberListStore.loadInitialData([USER_1, USER_2]);
  225. assigneeSelector.update();
  226. expect(assigneeSelector.find('LoadingIndicator').exists()).toBe(false);
  227. expect(
  228. assigneeSelector.find('InviteMemberLink[data-test-id="invite-member"]')
  229. ).toHaveLength(0);
  230. assigneeSelector.unmount();
  231. jest.spyOn(ConfigStore, 'get').mockImplementation(() => true);
  232. assigneeSelector = mountWithTheme(
  233. <AssigneeSelectorComponent id={GROUP_1.id} />,
  234. routerContext
  235. );
  236. await tick();
  237. assigneeSelector.update();
  238. openMenu();
  239. expect(
  240. assigneeSelector.find('InviteMemberLink[data-test-id="invite-member"]')
  241. ).toHaveLength(1);
  242. ConfigStore.get.mockRestore();
  243. });
  244. it('requires org:write to invite member', async function() {
  245. MemberListStore.loadInitialData([USER_1, USER_2]);
  246. jest.spyOn(ConfigStore, 'get').mockImplementation(() => true);
  247. // Remove org:write access permission and make sure invite member button is not shown.
  248. assigneeSelector.unmount();
  249. assigneeSelector = mountWithTheme(
  250. <AssigneeSelectorComponent id={GROUP_1.id} />,
  251. TestStubs.routerContext([{organization: TestStubs.Organization({access: []})}])
  252. );
  253. openMenu();
  254. assigneeSelector.update();
  255. expect(
  256. assigneeSelector.find('InviteMemberLink[data-test-id="invite-member"]')
  257. ).toHaveLength(0);
  258. ConfigStore.get.mockRestore();
  259. });
  260. it('filters user by email and selects with keyboard', async function() {
  261. openMenu();
  262. MemberListStore.loadInitialData([USER_1, USER_2]);
  263. assigneeSelector.update();
  264. expect(assigneeSelector.find('LoadingIndicator').exists()).toBe(false);
  265. assigneeSelector
  266. .find('StyledInput')
  267. .simulate('change', {target: {value: 'JohnSmith@example.com'}});
  268. expect(assigneeSelector.find('Avatar')).toHaveLength(1);
  269. expect(assigneeSelector.find('Avatar').prop('user')).toEqual(USER_2);
  270. assigneeSelector.find('StyledInput').simulate('keyDown', {key: 'Enter'});
  271. assigneeSelector.update();
  272. expect(assignMock).toHaveBeenLastCalledWith(
  273. '/issues/1337/',
  274. expect.objectContaining({
  275. data: {assignedTo: 'user:2'},
  276. })
  277. );
  278. expect(assigneeSelector.find('LoadingIndicator')).toHaveLength(1);
  279. await tick();
  280. await tick();
  281. assigneeSelector.update();
  282. expect(assigneeSelector.find('LoadingIndicator')).toHaveLength(0);
  283. expect(assigneeSelector.find('ActorAvatar')).toHaveLength(1);
  284. });
  285. });