organizationMemberRow.spec.jsx 8.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289
  1. import React from 'react';
  2. import {shallow} from 'enzyme';
  3. import OrganizationMemberRow from 'app/views/settings/organizationMembers/organizationMemberRow';
  4. const findWithText = (wrapper, text) =>
  5. wrapper.filterWhere(n => n.prop('children') && n.prop('children').includes(text));
  6. describe('OrganizationMemberRow', function() {
  7. let member = {
  8. id: '1',
  9. email: '',
  10. name: '',
  11. role: '',
  12. roleName: '',
  13. pending: false,
  14. flags: {
  15. 'sso:linked': false,
  16. },
  17. user: {
  18. id: '',
  19. has2fa: false,
  20. name: 'sentry@test.com',
  21. },
  22. };
  23. let currentUser = {
  24. id: '2',
  25. email: 'currentUser@email.com',
  26. };
  27. let defaultProps = {
  28. routes: [],
  29. orgId: 'org-slug',
  30. orgName: 'Organization Name',
  31. status: '',
  32. requireLink: false,
  33. memberCanLeave: false,
  34. canAddMembers: false,
  35. canRemoveMembers: false,
  36. member,
  37. currentUser,
  38. onSendInvite: () => {},
  39. onRemove: () => {},
  40. onLeave: () => {},
  41. };
  42. beforeEach(function() {});
  43. it('does not have 2fa warning if user has 2fa', function() {
  44. let wrapper = shallow(
  45. <OrganizationMemberRow
  46. {...defaultProps}
  47. member={{
  48. ...member,
  49. user: {
  50. ...member.user,
  51. has2fa: true,
  52. },
  53. }}
  54. />
  55. );
  56. expect(wrapper.find('NoTwoFactorIcon')).toHaveLength(0);
  57. expect(wrapper.find('HasTwoFactorIcon')).toHaveLength(1);
  58. });
  59. it('has 2fa warning if user does not have 2fa enabled', function() {
  60. let wrapper = shallow(
  61. <OrganizationMemberRow
  62. {...defaultProps}
  63. member={{
  64. ...member,
  65. user: {
  66. ...member.user,
  67. has2fa: false,
  68. },
  69. }}
  70. />
  71. );
  72. expect(wrapper.find('NoTwoFactorIcon')).toHaveLength(1);
  73. expect(wrapper.find('HasTwoFactorIcon')).toHaveLength(0);
  74. });
  75. describe('Pending user', function() {
  76. let props = {
  77. ...defaultProps,
  78. member: {
  79. ...member,
  80. pending: true,
  81. },
  82. };
  83. it('has "Invited" status, no "Resend Invite"', function() {
  84. let wrapper = shallow(
  85. <OrganizationMemberRow
  86. {...props}
  87. member={{
  88. ...member,
  89. pending: true,
  90. }}
  91. />
  92. );
  93. expect(findWithText(wrapper.find('strong'), 'Invited')).toHaveLength(1);
  94. expect(wrapper.find('ResendInviteButton')).toHaveLength(0);
  95. });
  96. it('has "Resend Invite" button only if `canAddMembers` is true', function() {
  97. let wrapper = shallow(<OrganizationMemberRow {...props} canAddMembers={true} />);
  98. expect(findWithText(wrapper.find('strong'), 'Invited')).toHaveLength(1);
  99. expect(wrapper.find('ResendInviteButton')).toHaveLength(1);
  100. });
  101. it('has the right inviting states', function() {
  102. let wrapper = shallow(<OrganizationMemberRow {...props} canAddMembers={true} />);
  103. expect(wrapper.find('ResendInviteButton')).toHaveLength(1);
  104. wrapper = shallow(
  105. <OrganizationMemberRow {...props} canAddMembers={true} status="loading" />
  106. );
  107. // Should have loader
  108. expect(wrapper.find('LoadingIndicator')).toHaveLength(1);
  109. // No Resend Invite button
  110. expect(wrapper.find('ResendInviteButton')).toHaveLength(0);
  111. wrapper = shallow(
  112. <OrganizationMemberRow {...props} canAddMembers={true} status="success" />
  113. );
  114. // Should have loader
  115. expect(wrapper.find('LoadingIndicator')).toHaveLength(0);
  116. // No Resend Invite button
  117. expect(wrapper.find('ResendInviteButton')).toHaveLength(0);
  118. expect(findWithText(wrapper.find('span'), 'Sent!')).toHaveLength(1);
  119. });
  120. });
  121. describe('Expired user', function() {
  122. it('has "Expired" status', function() {
  123. let wrapper = shallow(
  124. <OrganizationMemberRow
  125. {...defaultProps}
  126. member={{
  127. ...member,
  128. pending: true,
  129. expired: true,
  130. }}
  131. />
  132. );
  133. expect(findWithText(wrapper.find('strong'), 'Expired')).toHaveLength(1);
  134. expect(wrapper.find('ResendInviteButton')).toHaveLength(0);
  135. });
  136. });
  137. describe('Requires SSO Link', function() {
  138. let props = {
  139. ...defaultProps,
  140. flags: {
  141. 'sso:link': false,
  142. },
  143. requireLink: true,
  144. };
  145. it('shows "Invited" status if user has not registered and not linked', function() {
  146. let wrapper = shallow(
  147. <OrganizationMemberRow
  148. {...props}
  149. member={{
  150. ...member,
  151. pending: true,
  152. }}
  153. />
  154. );
  155. expect(findWithText(wrapper.find('strong'), 'Invited')).toHaveLength(1);
  156. expect(wrapper.find('ResendInviteButton')).toHaveLength(0);
  157. });
  158. it('shows "missing SSO link" message if user is registered and needs link', function() {
  159. let wrapper = shallow(
  160. <OrganizationMemberRow
  161. {...props}
  162. member={{
  163. ...member,
  164. }}
  165. />
  166. );
  167. expect(findWithText(wrapper.find('strong'), 'Invited')).toHaveLength(0);
  168. expect(findWithText(wrapper.find('strong'), 'Missing SSO Link')).toHaveLength(1);
  169. expect(wrapper.find('ResendInviteButton')).toHaveLength(0);
  170. });
  171. it('has "Resend Invite" button only if `canAddMembers` is true and no link', function() {
  172. let wrapper = shallow(
  173. <OrganizationMemberRow
  174. {...props}
  175. canAddMembers={true}
  176. member={{
  177. ...member,
  178. }}
  179. />
  180. );
  181. expect(wrapper.find('ResendInviteButton')).toHaveLength(1);
  182. });
  183. it('has 2fa warning if user is linked does not have 2fa enabled', function() {
  184. let wrapper = shallow(
  185. <OrganizationMemberRow
  186. {...defaultProps}
  187. member={{
  188. ...member,
  189. flags: {
  190. 'sso:linked': true,
  191. },
  192. user: {
  193. ...member.user,
  194. has2fa: false,
  195. },
  196. }}
  197. />
  198. );
  199. expect(wrapper.find('NoTwoFactorIcon')).toHaveLength(1);
  200. expect(wrapper.find('HasTwoFactorIcon')).toHaveLength(0);
  201. });
  202. });
  203. describe('Is Current User', function() {
  204. let props = {
  205. ...defaultProps,
  206. member: {
  207. ...member,
  208. email: 'currentUser@email.com',
  209. },
  210. };
  211. it('has button to leave organization and no button to remove', function() {
  212. let wrapper = shallow(<OrganizationMemberRow {...props} memberCanLeave={true} />);
  213. expect(findWithText(wrapper.find('Button'), 'Leave')).toHaveLength(1);
  214. expect(findWithText(wrapper.find('Button'), 'Remove')).toHaveLength(0);
  215. });
  216. it('has disabled button to leave organization and no button to remove when member can not leave', function() {
  217. let wrapper = shallow(<OrganizationMemberRow {...props} memberCanLeave={false} />);
  218. expect(findWithText(wrapper.find('Button'), 'Leave')).toHaveLength(1);
  219. expect(
  220. findWithText(wrapper.find('Button'), 'Leave')
  221. .first()
  222. .prop('disabled')
  223. ).toBe(true);
  224. expect(findWithText(wrapper.find('Button'), 'Remove')).toHaveLength(0);
  225. });
  226. });
  227. describe('Not Current User', function() {
  228. let props = {
  229. ...defaultProps,
  230. };
  231. it('does not have Leave button', function() {
  232. let wrapper = shallow(<OrganizationMemberRow {...props} memberCanLeave={true} />);
  233. expect(findWithText(wrapper.find('Button'), 'Leave')).toHaveLength(0);
  234. });
  235. it('has Remove disabled button when `canRemoveMembers` is false', function() {
  236. let wrapper = shallow(<OrganizationMemberRow {...props} />);
  237. expect(findWithText(wrapper.find('Button'), 'Remove')).toHaveLength(1);
  238. expect(findWithText(wrapper.find('Button'), 'Remove').prop('disabled')).toBe(true);
  239. });
  240. it('has Remove button when `canRemoveMembers` is true', function() {
  241. let wrapper = shallow(<OrganizationMemberRow {...props} canRemoveMembers={true} />);
  242. let removeButton = findWithText(wrapper.find('Button'), 'Remove');
  243. expect(removeButton).toHaveLength(1);
  244. expect(removeButton.first().prop('disabled')).toBe(false);
  245. });
  246. });
  247. });