import {Organization} from 'sentry-fixture/organization'; import {initializeOrg} from 'sentry-test/initializeOrg'; import { render, renderGlobalModal, screen, userEvent, waitFor, } from 'sentry-test/reactTestingLibrary'; import SentryAppDetailedView from 'sentry/views/settings/organizationIntegrations/sentryAppDetailedView'; describe('SentryAppDetailedView', function () { const org = Organization(); const {router} = initializeOrg({ projects: [ {isMember: true, isBookmarked: true}, {isMember: true, slug: 'new-project', id: '3'}, ], organization: { features: ['events'], }, router: { location: { pathname: '/organizations/org-slug/events/', query: {}, }, }, }); afterEach(() => { MockApiClient.clearMockResponses(); jest.clearAllMocks(); }); describe('Published Sentry App', function () { let createRequest; let deleteRequest; let sentryAppInteractionRequest; beforeEach(() => { sentryAppInteractionRequest = MockApiClient.addMockResponse({ url: `/sentry-apps/clickup/interaction/`, method: 'POST', statusCode: 200, body: {}, }); MockApiClient.addMockResponse({ url: '/sentry-apps/clickup/', body: { status: 'published', scopes: [], isAlertable: false, clientSecret: '193583e573d14d61832de96a9efc32ceb64e59a494284f58b50328a656420a55', overview: null, verifyInstall: false, owner: {id: 1, slug: 'sentry'}, slug: 'clickup', name: 'ClickUp', uuid: '5d547ecb-7eb8-4ed2-853b-40256177d526', author: 'Nisanthan', webhookUrl: 'http://localhost:7000', clientId: 'c215db1accc040919e0b0dce058e0ecf4ea062bb82174d70aee8eba62351be24', redirectUrl: null, allowedOrigins: [], events: [], schema: {}, }, }); MockApiClient.addMockResponse({ url: '/sentry-apps/clickup/features/', body: [ { featureGate: 'integrations-api', description: 'ClickUp can **utilize the Sentry API** to pull data or update resources in Sentry (with permissions granted, of course).', }, ], }); MockApiClient.addMockResponse({ url: `/organizations/${org.slug}/sentry-app-installations/`, body: [], }); createRequest = MockApiClient.addMockResponse({ url: `/organizations/${org.slug}/sentry-app-installations/`, body: { status: 'installed', organization: {slug: `${org.slug}`}, app: {uuid: '5d547ecb-7eb8-4ed2-853b-40256177d526', slug: 'clickup'}, code: '1dc8b0a28b7f45959d01bbc99d9bd568', uuid: '687323fd-9fa4-4f8f-9bee-ca0089224b3e', }, method: 'POST', }); deleteRequest = MockApiClient.addMockResponse({ url: '/sentry-app-installations/687323fd-9fa4-4f8f-9bee-ca0089224b3e/', body: {}, method: 'DELETE', }); }); it('renders a published sentry app', () => { render( ); expect(sentryAppInteractionRequest).toHaveBeenCalledWith( `/sentry-apps/clickup/interaction/`, expect.objectContaining({ method: 'POST', data: { tsdbField: 'sentry_app_viewed', }, }) ); // Shows the Integration name and install status expect(screen.getByText('ClickUp')).toBeInTheDocument(); expect(screen.getByText('Not Installed')).toBeInTheDocument(); // Shows the Accept & Install button expect(screen.getByRole('button', {name: 'Accept & Install'})).toBeEnabled(); }); it('installs and uninstalls', async function () { render( ); renderGlobalModal(); await userEvent.click(screen.getByRole('button', {name: 'Accept & Install'})); expect(createRequest).toHaveBeenCalledTimes(1); expect(await screen.findByRole('button', {name: 'Uninstall'})).toBeInTheDocument(); await userEvent.click(screen.getByRole('button', {name: 'Uninstall'})); await userEvent.click(screen.getByRole('button', {name: 'Confirm'})); expect(deleteRequest).toHaveBeenCalledTimes(1); }); }); describe('Internal Sentry App', function () { beforeEach(() => { MockApiClient.addMockResponse({ url: `/sentry-apps/my-headband-washer-289499/interaction/`, method: 'POST', statusCode: 200, body: {}, }); MockApiClient.addMockResponse({ url: '/sentry-apps/my-headband-washer-289499/', body: { status: 'internal', scopes: [ 'project:read', 'team:read', 'team:write', 'project:releases', 'event:read', 'org:read', 'member:read', 'member:write', ], isAlertable: false, clientSecret: '8f47dcef40f7486f9bacfeca257022e092a483add7cf4d619993b9ace9775a79', overview: null, verifyInstall: false, owner: {id: 1, slug: 'sentry'}, slug: 'my-headband-washer-289499', name: 'My Headband Washer', uuid: 'a806ab10-9608-4a4f-8dd9-ca6d6c09f9f5', author: 'Sentry', webhookUrl: 'https://myheadbandwasher.com', clientId: 'a6d35972d4164ef18845b1e2ca954fe70ac196e0b20d4d1e8760a38772cf6f1c', redirectUrl: null, allowedOrigins: [], events: [], schema: {}, }, }); MockApiClient.addMockResponse({ url: '/sentry-apps/my-headband-washer-289499/features/', body: [ { featureGate: 'integrations-api', description: 'My Headband Washer can **utilize the Sentry API** to pull data or update resources in Sentry (with permissions granted, of course).', }, ], }); MockApiClient.addMockResponse({ url: `/organizations/${org.slug}/sentry-app-installations/`, body: [], }); }); it('should get redirected to Developer Settings', () => { render( ); expect(router.push).toHaveBeenLastCalledWith( `/settings/${org.slug}/developer-settings/my-headband-washer-289499/` ); }); }); describe('Unpublished Sentry App without Redirect Url', function () { let createRequest; beforeEach(() => { MockApiClient.addMockResponse({ url: `/sentry-apps/la-croix-monitor/interaction/`, method: 'POST', statusCode: 200, body: {}, }); MockApiClient.addMockResponse({ url: '/sentry-apps/la-croix-monitor/', body: { status: 'unpublished', scopes: [ 'project:read', 'project:write', 'team:read', 'project:releases', 'event:read', 'org:read', ], isAlertable: false, clientSecret: '2b2aeb743c3745ab832e03bf02a7d91851908d379646499f900cd115780e8b2b', overview: null, verifyInstall: false, owner: {id: 1, slug: 'sentry'}, slug: 'la-croix-monitor', name: 'La Croix Monitor', uuid: 'a59c8fcc-2f27-49f8-af9e-02661fc3e8d7', author: 'La Croix', webhookUrl: 'https://lacroix.com', clientId: '8cc36458a0f94c93816e06dce7d808f882cbef59af6040d2b9ec4d67092c80f1', redirectUrl: null, allowedOrigins: [], events: [], schema: {}, }, }); MockApiClient.addMockResponse({ url: '/sentry-apps/la-croix-monitor/features/', body: [ { featureGate: 'integrations-api', description: 'La Croix Monitor can **utilize the Sentry API** to pull data or update resources in Sentry (with permissions granted, of course).', }, ], }); MockApiClient.addMockResponse({ url: `/organizations/${org.slug}/sentry-app-installations/`, body: [], }); createRequest = MockApiClient.addMockResponse({ url: `/organizations/${org.slug}/sentry-app-installations/`, method: 'POST', body: { status: 'installed', organization: {slug: 'sentry'}, app: {uuid: 'a59c8fcc-2f27-49f8-af9e-02661fc3e8d7', slug: 'la-croix-monitor'}, code: '21c87231918a4e5c85d9b9e799c07382', uuid: '258ad77c-7e6c-4cfe-8a40-6171cff30d61', }, }); }); it('shows the Integration name and install status', function () { render( ); expect(screen.getByText('La Croix Monitor')).toBeInTheDocument(); expect(screen.getByText('Not Installed')).toBeInTheDocument(); }); it('installs and uninstalls', async function () { render( ); renderGlobalModal(); await userEvent.click(screen.getByRole('button', {name: 'Accept & Install'})); expect(createRequest).toHaveBeenCalledTimes(1); }); }); describe('Unpublished Sentry App with Redirect Url', function () { let createRequest; beforeEach(() => { MockApiClient.addMockResponse({ url: `/sentry-apps/go-to-google/interaction/`, method: 'POST', statusCode: 200, body: {}, }); MockApiClient.addMockResponse({ url: '/sentry-apps/go-to-google/', body: { status: 'unpublished', scopes: ['project:read', 'team:read'], isAlertable: false, clientSecret: '6405a4a7b8084cdf8dbea53b53e2163983deb428b78e4c6997bc408d44d93878', overview: null, verifyInstall: false, owner: {id: 1, slug: 'sentry'}, slug: 'go-to-google', name: 'Go to Google', uuid: 'a4b8f364-4300-41ac-b8af-d8791ad50e77', author: 'Nisanthan Nanthakumar', webhookUrl: 'https://www.google.com', clientId: '0974b5df6b57480b99c2e1f238eef769ef2c27ec156d4791a26903a896d5807e', redirectUrl: 'https://www.google.com', allowedOrigins: [], events: [], schema: {}, }, }); MockApiClient.addMockResponse({ url: '/sentry-apps/go-to-google/features/', body: [ { featureGate: 'integrations-api', description: 'Go to Google can **utilize the Sentry API** to pull data or update resources in Sentry (with permissions granted, of course).', }, ], }); MockApiClient.addMockResponse({ url: `/organizations/${org.slug}/sentry-app-installations/`, body: [], }); createRequest = MockApiClient.addMockResponse({ url: `/organizations/${org.slug}/sentry-app-installations/`, body: { status: 'installed', organization: {slug: 'sentry'}, app: {uuid: 'a4b8f364-4300-41ac-b8af-d8791ad50e77', slug: 'go-to-google'}, code: '1f0e7c1b99b940abac7a19b86e69bbe1', uuid: '4d803538-fd42-4278-b410-492f5ab677b5', }, method: 'POST', }); }); it('shows the Integration name and install status', function () { render( ); expect(screen.getByText('Go to Google')).toBeInTheDocument(); expect(screen.getByText('Not Installed')).toBeInTheDocument(); // Shows the Accept & Install button expect(screen.getByRole('button', {name: 'Accept & Install'})).toBeEnabled(); }); it('onClick: redirects url', async function () { render( ); const locationAssignSpy = jest.spyOn(window.location, 'assign'); await userEvent.click(screen.getByRole('button', {name: 'Accept & Install'})); expect(createRequest).toHaveBeenCalled(); await waitFor(() => { expect(locationAssignSpy).toHaveBeenLastCalledWith( 'https://www.google.com/?code=1f0e7c1b99b940abac7a19b86e69bbe1&installationId=4d803538-fd42-4278-b410-492f5ab677b5&orgSlug=org-slug' ); }); }); }); });