integrationRepos.spec.jsx 5.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187
  1. import {mountWithTheme} from 'sentry-test/enzyme';
  2. import RepositoryActions from 'sentry/actions/repositoryActions';
  3. import {Client} from 'sentry/api';
  4. import IntegrationRepos from 'sentry/views/organizationIntegrations/integrationRepos';
  5. describe('IntegrationRepos', function () {
  6. const org = TestStubs.Organization();
  7. const integration = TestStubs.GitHubIntegration();
  8. beforeEach(() => {
  9. Client.clearMockResponses();
  10. jest.spyOn(RepositoryActions, 'resetRepositories');
  11. });
  12. afterEach(() => {
  13. jest.restoreAllMocks();
  14. });
  15. describe('Getting repositories', function () {
  16. it('handles broken integrations', function () {
  17. Client.addMockResponse({
  18. url: `/organizations/${org.slug}/integrations/1/repos/`,
  19. statusCode: 400,
  20. body: {detail: 'Invalid grant'},
  21. });
  22. Client.addMockResponse({
  23. url: `/organizations/${org.slug}/repos/`,
  24. method: 'GET',
  25. body: [],
  26. });
  27. const wrapper = mountWithTheme(<IntegrationRepos integration={integration} />);
  28. expect(wrapper.find('PanelBody')).toHaveLength(0);
  29. expect(wrapper.find('Alert')).toHaveLength(1);
  30. });
  31. });
  32. describe('Adding repositories', function () {
  33. it('can save successfully', async function () {
  34. const addRepo = Client.addMockResponse({
  35. url: `/organizations/${org.slug}/repos/`,
  36. method: 'POST',
  37. body: TestStubs.Repository({integrationId: '1'}),
  38. });
  39. Client.addMockResponse({
  40. url: `/organizations/${org.slug}/integrations/1/repos/`,
  41. body: {
  42. repos: [{identifier: 'example/repo-name', name: 'repo-name'}],
  43. },
  44. });
  45. Client.addMockResponse({
  46. url: `/organizations/${org.slug}/repos/`,
  47. method: 'GET',
  48. body: [],
  49. });
  50. const wrapper = mountWithTheme(<IntegrationRepos integration={integration} />);
  51. wrapper.find('DropdownButton').simulate('click');
  52. wrapper.find('StyledListElement').simulate('click');
  53. expect(addRepo).toHaveBeenCalledWith(
  54. `/organizations/${org.slug}/repos/`,
  55. expect.objectContaining({
  56. data: {
  57. installation: '1',
  58. provider: 'integrations:github',
  59. identifier: 'example/repo-name',
  60. },
  61. })
  62. );
  63. await tick();
  64. wrapper.update();
  65. const name = wrapper.find('RepositoryRow').find('strong').first();
  66. expect(name).toHaveLength(1);
  67. expect(name.text()).toEqual('example/repo-name');
  68. expect(RepositoryActions.resetRepositories).toHaveBeenCalled();
  69. });
  70. it('handles failure during save', function () {
  71. const addRepo = Client.addMockResponse({
  72. url: `/organizations/${org.slug}/repos/`,
  73. method: 'POST',
  74. statusCode: 400,
  75. body: {
  76. errors: {
  77. __all__: 'Repository already exists.',
  78. },
  79. },
  80. });
  81. Client.addMockResponse({
  82. url: `/organizations/${org.slug}/integrations/1/repos/`,
  83. body: {
  84. repos: [{identifier: 'getsentry/sentry', name: 'sentry'}],
  85. },
  86. });
  87. Client.addMockResponse({
  88. url: `/organizations/${org.slug}/repos/`,
  89. method: 'GET',
  90. body: [],
  91. });
  92. const wrapper = mountWithTheme(<IntegrationRepos integration={integration} />);
  93. wrapper.find('DropdownButton').simulate('click');
  94. wrapper.find('StyledListElement').simulate('click');
  95. wrapper.update();
  96. expect(addRepo).toHaveBeenCalled();
  97. expect(wrapper.find('RepoOption')).toHaveLength(0);
  98. });
  99. });
  100. describe('migratable repo', function () {
  101. it('associates repository with integration', async () => {
  102. Client.addMockResponse({
  103. url: `/organizations/${org.slug}/repos/`,
  104. body: [
  105. TestStubs.Repository({
  106. integrationId: null,
  107. externalSlug: 'example/repo-name',
  108. provider: {
  109. id: 'integrations:github',
  110. name: 'GitHub',
  111. status: 'active',
  112. },
  113. }),
  114. ],
  115. });
  116. Client.addMockResponse({
  117. url: `/organizations/${org.slug}/integrations/${integration.id}/repos/`,
  118. body: {repos: [{identifier: 'example/repo-name', name: 'repo-name'}]},
  119. });
  120. const updateRepo = Client.addMockResponse({
  121. method: 'PUT',
  122. url: `/organizations/${org.slug}/repos/4/`,
  123. body: {},
  124. });
  125. const wrapper = mountWithTheme(<IntegrationRepos integration={integration} />);
  126. wrapper.find('DropdownButton').simulate('click');
  127. wrapper.find('StyledListElement').simulate('click');
  128. await tick();
  129. expect(updateRepo).toHaveBeenCalledWith(
  130. `/organizations/${org.slug}/repos/4/`,
  131. expect.objectContaining({
  132. data: {integrationId: '1'},
  133. })
  134. );
  135. expect(RepositoryActions.resetRepositories).toHaveBeenCalled();
  136. });
  137. it('uses externalSlug not name for comparison', () => {
  138. Client.addMockResponse({
  139. url: `/organizations/${org.slug}/repos/`,
  140. method: 'GET',
  141. body: [TestStubs.Repository({name: 'repo-name', externalSlug: 9876})],
  142. });
  143. const getItems = Client.addMockResponse({
  144. url: `/organizations/${org.slug}/integrations/${integration.id}/repos/`,
  145. method: 'GET',
  146. body: {
  147. repos: [{identifier: 9876, name: 'repo-name'}],
  148. },
  149. });
  150. const updateRepo = Client.addMockResponse({
  151. method: 'PUT',
  152. url: `/organizations/${org.slug}/repos/4/`,
  153. body: {},
  154. });
  155. const wrapper = mountWithTheme(<IntegrationRepos integration={integration} />);
  156. wrapper.find('DropdownButton').simulate('click');
  157. wrapper.find('StyledListElement').simulate('click');
  158. expect(getItems).toHaveBeenCalled();
  159. expect(updateRepo).toHaveBeenCalledWith(
  160. `/organizations/${org.slug}/repos/4/`,
  161. expect.objectContaining({
  162. data: {integrationId: '1'},
  163. })
  164. );
  165. });
  166. });
  167. });