index.spec.tsx 8.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240
  1. import {Organization} from 'fixtures/js-stubs/organization';
  2. import {Project} from 'fixtures/js-stubs/project';
  3. import {SentryApp} from 'fixtures/js-stubs/sentryApp';
  4. import {Team} from 'fixtures/js-stubs/team';
  5. import {render, screen} from 'sentry-test/reactTestingLibrary';
  6. import AvatarComponent from 'sentry/components/avatar';
  7. import ConfigStore from 'sentry/stores/configStore';
  8. import {Avatar} from 'sentry/types';
  9. describe('Avatar', function () {
  10. const avatar: Avatar = {
  11. avatarType: 'gravatar',
  12. avatarUuid: '2d641b5d-8c74-44de-9cb6-fbd54701b35e',
  13. };
  14. const user = {
  15. id: '1',
  16. name: 'Jane Bloggs',
  17. username: 'janebloggs@example.com',
  18. email: 'janebloggs@example.com',
  19. ip_address: '127.0.0.1',
  20. avatar,
  21. };
  22. const userNameInitials = user.name
  23. .split(' ')
  24. .map(n => n[0])
  25. .join('');
  26. describe('render()', function () {
  27. it('has `avatar` className', async function () {
  28. render(<AvatarComponent user={user} />);
  29. await screen.findByRole('img');
  30. const avatarElement = screen.getByTestId(`${avatar.avatarType}-avatar`);
  31. expect(avatarElement).toBeInTheDocument();
  32. expect(avatarElement).toHaveAttribute('title', user.name);
  33. });
  34. it('should show a gravatar when avatar type is gravatar', async function () {
  35. const gravatarBaseUrl = 'gravatarBaseUrl';
  36. ConfigStore.set('gravatarBaseUrl', gravatarBaseUrl);
  37. render(<AvatarComponent user={user} />);
  38. expect(screen.getByTestId(`${avatar.avatarType}-avatar`)).toBeInTheDocument();
  39. const avatarImage = await screen.findByRole('img');
  40. expect(avatarImage).toHaveAttribute(
  41. 'src',
  42. `${gravatarBaseUrl}/avatar/a94c88e18c44e553497bf642449b6398?d=404&s=120`
  43. );
  44. });
  45. it('should show an upload when avatar type is upload', async function () {
  46. avatar.avatarType = 'upload';
  47. render(<AvatarComponent user={user} />);
  48. expect(screen.getByTestId(`${avatar.avatarType}-avatar`)).toBeInTheDocument();
  49. const avatarImage = await screen.findByRole('img');
  50. expect(avatarImage).toHaveAttribute('src', `/avatar/${avatar.avatarUuid}/?s=120`);
  51. });
  52. it('should show an upload with the correct size (static 120 size)', async function () {
  53. const avatar1 = render(<AvatarComponent user={user} size={76} />);
  54. expect(await screen.findByRole('img')).toHaveAttribute(
  55. 'src',
  56. `/avatar/${avatar.avatarUuid}/?s=120`
  57. );
  58. avatar1.unmount();
  59. const avatar2 = render(<AvatarComponent user={user} size={121} />);
  60. expect(await screen.findByRole('img')).toHaveAttribute(
  61. 'src',
  62. `/avatar/${avatar.avatarUuid}/?s=120`
  63. );
  64. avatar2.unmount();
  65. const avatar3 = render(<AvatarComponent user={user} size={32} />);
  66. expect(await screen.findByRole('img')).toHaveAttribute(
  67. 'src',
  68. `/avatar/${avatar.avatarUuid}/?s=120`
  69. );
  70. avatar3.unmount();
  71. render(<AvatarComponent user={user} size={1} />);
  72. expect(await screen.findByRole('img')).toHaveAttribute(
  73. 'src',
  74. `/avatar/${avatar.avatarUuid}/?s=120`
  75. );
  76. });
  77. it('should not show upload or gravatar when avatar type is letter', function () {
  78. avatar.avatarType = 'letter_avatar';
  79. render(<AvatarComponent user={user} />);
  80. expect(screen.getByTestId(`${avatar.avatarType}-avatar`)).toBeInTheDocument();
  81. expect(screen.getByText(userNameInitials)).toBeInTheDocument();
  82. });
  83. it('use letter avatar by default, when no avatar type is set and user has an email address', function () {
  84. render(<AvatarComponent user={{...user, avatar: undefined}} />);
  85. expect(screen.getByTestId(`${avatar.avatarType}-avatar`)).toBeInTheDocument();
  86. expect(screen.getByText(userNameInitials)).toBeInTheDocument();
  87. });
  88. it('should show a gravatar when no avatar type is set and user has an email address', async function () {
  89. render(<AvatarComponent gravatar user={{...user, avatar: undefined}} />);
  90. await screen.findByRole('img');
  91. const avatarElement = screen.getByTestId(`gravatar-avatar`);
  92. expect(avatarElement).toBeInTheDocument();
  93. expect(avatarElement).toHaveAttribute('title', user.name);
  94. });
  95. it('should not show a gravatar when no avatar type is set and user has no email address', function () {
  96. render(<AvatarComponent gravatar user={{...user, email: '', avatar: undefined}} />);
  97. expect(screen.getByTestId(`letter_avatar-avatar`)).toBeInTheDocument();
  98. expect(screen.getByText(userNameInitials)).toBeInTheDocument();
  99. });
  100. it('can display a team Avatar', function () {
  101. const team = Team({slug: 'test-team_test'});
  102. render(<AvatarComponent team={team} />);
  103. expect(screen.getByTestId(`letter_avatar-avatar`)).toBeInTheDocument();
  104. expect(screen.getByText('TT')).toBeInTheDocument();
  105. });
  106. it('can display an organization Avatar', function () {
  107. const organization = Organization({slug: 'test-organization'});
  108. render(<AvatarComponent organization={organization} />);
  109. expect(screen.getByTestId(`letter_avatar-avatar`)).toBeInTheDocument();
  110. expect(screen.getByText('TO')).toBeInTheDocument();
  111. });
  112. it('displays platform list icons for project Avatar', function () {
  113. const project = Project({
  114. platforms: ['python', 'javascript'],
  115. platform: 'java',
  116. });
  117. render(<AvatarComponent project={project} />);
  118. const platformIcon = screen.getByRole('img');
  119. expect(platformIcon).toBeInTheDocument();
  120. expect(platformIcon).toHaveAttribute(
  121. 'data-test-id',
  122. `platform-icon-${project.platform}`
  123. );
  124. });
  125. it('displays a fallback platform list for project Avatar using the `platform` specified during onboarding', function () {
  126. const project = Project({platform: 'java'});
  127. render(<AvatarComponent project={project} />);
  128. const platformIcon = screen.getByRole('img');
  129. expect(platformIcon).toBeInTheDocument();
  130. expect(platformIcon).toHaveAttribute(
  131. 'data-test-id',
  132. `platform-icon-${project.platform}`
  133. );
  134. });
  135. it('uses onboarding project when platforms is an empty array', function () {
  136. const project = Project({platforms: [], platform: 'java'});
  137. render(<AvatarComponent project={project} />);
  138. const platformIcon = screen.getByRole('img');
  139. expect(platformIcon).toBeInTheDocument();
  140. expect(platformIcon).toHaveAttribute(
  141. 'data-test-id',
  142. `platform-icon-${project.platform}`
  143. );
  144. });
  145. it('renders the correct SentryApp depending on its props', async function () {
  146. const colorAvatar = {avatarUuid: 'abc', avatarType: 'upload', color: true};
  147. const simpleAvatar = {avatarUuid: 'def', avatarType: 'upload', color: false};
  148. const sentryApp = SentryApp({
  149. avatars: [colorAvatar, simpleAvatar],
  150. });
  151. const avatar1 = render(<AvatarComponent sentryApp={sentryApp} isColor />);
  152. expect(await screen.findByRole('img')).toHaveAttribute(
  153. 'src',
  154. `/sentry-app-avatar/${colorAvatar.avatarUuid}/?s=120`
  155. );
  156. avatar1.unmount();
  157. const avatar2 = render(<AvatarComponent sentryApp={sentryApp} isColor={false} />);
  158. expect(await screen.findByRole('img')).toHaveAttribute(
  159. 'src',
  160. `/sentry-app-avatar/${simpleAvatar.avatarUuid}/?s=120`
  161. );
  162. avatar2.unmount();
  163. render(<AvatarComponent sentryApp={sentryApp} isDefault />);
  164. expect(screen.getByTestId('default-sentry-app-avatar')).toBeInTheDocument();
  165. });
  166. it('renders the correct fallbacks for SentryAppAvatars', async function () {
  167. const colorAvatar = {avatarUuid: 'abc', avatarType: 'upload', color: true};
  168. const sentryApp = SentryApp({avatars: []});
  169. // No existing avatars
  170. const avatar1 = render(<AvatarComponent sentryApp={sentryApp} isColor />);
  171. expect(screen.getByTestId('default-sentry-app-avatar')).toBeInTheDocument();
  172. avatar1.unmount();
  173. // No provided `isColor` attribute
  174. sentryApp.avatars.push(colorAvatar);
  175. const avatar2 = render(<AvatarComponent sentryApp={sentryApp} />);
  176. expect(await screen.findByRole('img')).toHaveAttribute(
  177. 'src',
  178. `/sentry-app-avatar/${colorAvatar.avatarUuid}/?s=120`
  179. );
  180. avatar2.unmount();
  181. // avatarType of `default`
  182. sentryApp.avatars[0].avatarType = 'default';
  183. render(<AvatarComponent sentryApp={sentryApp} isColor />);
  184. expect(screen.getByTestId('default-sentry-app-avatar')).toBeInTheDocument();
  185. });
  186. });
  187. });