import selectEvent from 'react-select-event'; import pick from 'lodash/pick'; import {SentryApp} from 'sentry-fixture/sentryApp'; import {render, screen, userEvent, waitFor} from 'sentry-test/reactTestingLibrary'; import {textWithMarkupMatcher} from 'sentry-test/utils'; import type {Organization} from 'sentry/types'; import SentryAppExternalInstallation from 'sentry/views/sentryAppExternalInstallation'; describe('SentryAppExternalInstallation', () => { let sentryApp: ReturnType, getOrgsMock: ReturnType, getOrgMock: ReturnType, getAppMock: ReturnType, getInstallationsMock: ReturnType, getFeaturesMock: ReturnType, org1: Organization, org1Lite: Pick, org2: Organization, org2Lite: Pick; beforeEach(() => { MockApiClient.clearMockResponses(); org1 = TestStubs.Organization({ slug: 'org1', name: 'Organization 1', }); org2 = TestStubs.Organization({ slug: 'org2', name: 'Organization 2', }); org1Lite = pick(org1, ['slug', 'name', 'id']); org2Lite = pick(org2, ['slug', 'name', 'id']); sentryApp = SentryApp({ status: 'published', redirectUrl: 'https://google.com', }); getAppMock = MockApiClient.addMockResponse({ url: `/sentry-apps/${sentryApp.slug}/`, body: sentryApp, }); getFeaturesMock = MockApiClient.addMockResponse({ url: `/sentry-apps/${sentryApp.slug}/features/`, body: [], }); MockApiClient.addMockResponse({ url: `/sentry-apps/${sentryApp.slug}/interaction/`, method: 'POST', statusCode: 200, body: {}, }); }); describe('single organization', () => { beforeEach(() => { getOrgsMock = MockApiClient.addMockResponse({ url: '/organizations/', body: [org1Lite], }); getOrgMock = MockApiClient.addMockResponse({ url: `/organizations/${org1.slug}/`, body: org1, }); getInstallationsMock = MockApiClient.addMockResponse({ url: `/organizations/${org1.slug}/sentry-app-installations/`, body: [], }); }); it('sets the org automatically', () => { render( ); expect(getAppMock).toHaveBeenCalled(); expect(getOrgsMock).toHaveBeenCalled(); expect(getOrgMock).toHaveBeenCalled(); expect(getInstallationsMock).toHaveBeenCalled(); expect( screen.getByText( textWithMarkupMatcher( 'You are installing Sample App for organization Organization 1' ) ) ).toBeInTheDocument(); expect(screen.queryByText('Select an organization')).not.toBeInTheDocument(); }); it('installs and redirects', async () => { const installUrl = `/organizations/${org1.slug}/sentry-app-installations/`; const install = { uuid: 'fake-id', code: 'some-code', }; const installMock = MockApiClient.addMockResponse({ url: installUrl, method: 'POST', body: install, }); render( ); await userEvent.click(await screen.findByTestId('install')); expect(installMock).toHaveBeenCalledWith( installUrl, expect.objectContaining({ data: {slug: sentryApp.slug}, }) ); await waitFor(() => { expect(window.location.assign).toHaveBeenCalledWith( `https://google.com/?code=${install.code}&installationId=${install.uuid}&orgSlug=${org1.slug}` ); }); (window.location.assign as jest.Mock).mockClear(); }); }); describe('multiple organizations', () => { beforeEach(() => { getOrgsMock = MockApiClient.addMockResponse({ url: '/organizations/', body: [org1Lite, org2Lite], }); }); it('renders org dropdown', () => { render( ); expect(getAppMock).toHaveBeenCalled(); expect(getOrgsMock).toHaveBeenCalled(); expect(screen.getByText('Select an organization')).toBeInTheDocument(); }); it('selecting org from dropdown loads the org through the API', async () => { getOrgMock = MockApiClient.addMockResponse({ url: `/organizations/${org2.slug}/`, body: org2, }); getInstallationsMock = MockApiClient.addMockResponse({ url: `/organizations/${org2.slug}/sentry-app-installations/`, body: [], }); render( ); await selectEvent.select(screen.getByText('Select an organization'), 'org2'); expect(getOrgMock).toHaveBeenCalledTimes(1); expect(getOrgMock).toHaveBeenLastCalledWith( '/organizations/org2/', expect.anything() ); expect(getInstallationsMock).toHaveBeenCalled(); expect(getFeaturesMock).toHaveBeenCalled(); await waitFor(() => expect(screen.getByTestId('install')).toBeEnabled()); }); }); });