import {initializeOrg} from 'sentry-test/initializeOrg'; import { render, renderGlobalModal, screen, userEvent, waitFor, } from 'sentry-test/reactTestingLibrary'; import {ProjectSourceMaps} from 'sentry/views/settings/projectSourceMaps/projectSourceMaps'; function renderReleaseBundlesMockRequests({ orgSlug, projectSlug, empty, }: { orgSlug: string; projectSlug: string; empty?: boolean; }) { const sourceMaps = MockApiClient.addMockResponse({ url: `/projects/${orgSlug}/${projectSlug}/files/source-maps/`, body: empty ? [] : [ TestStubs.SourceMapArchive(), TestStubs.SourceMapArchive({ id: 2, name: 'abc', fileCount: 3, date: '2023-05-06T13:41:00Z', }), ], }); return {sourceMaps}; } function renderDebugIdBundlesMockRequests({ orgSlug, projectSlug, empty, }: { orgSlug: string; projectSlug: string; empty?: boolean; }) { const artifactBundles = MockApiClient.addMockResponse({ url: `/projects/${orgSlug}/${projectSlug}/files/artifact-bundles/`, body: empty ? [] : TestStubs.SourceMapsDebugIDBundles(), }); const artifactBundlesDeletion = MockApiClient.addMockResponse({ url: `/projects/${orgSlug}/${projectSlug}/files/artifact-bundles/`, method: 'DELETE', }); return {artifactBundles, artifactBundlesDeletion}; } describe('ProjectSourceMaps', function () { describe('Release Bundles', function () { it('renders default state', async function () { const {organization, route, project, router, routerContext} = initializeOrg({ router: { location: { query: {}, pathname: `/settings/${initializeOrg().organization.slug}/projects/${ initializeOrg().project.slug }/source-maps/release-bundles/`, }, params: {}, }, }); const mockRequests = renderReleaseBundlesMockRequests({ orgSlug: organization.slug, projectSlug: project.slug, }); render( , {context: routerContext, organization} ); // Title expect(screen.getByRole('heading', {name: 'Source Maps'})).toBeInTheDocument(); // Active tab const tabs = screen.getAllByRole('listitem'); expect(tabs).toHaveLength(2); // Tab 1 expect(tabs[0]).toHaveTextContent('Artifact Bundles'); expect(tabs[0]).not.toHaveClass('active'); // Tab 2 expect(tabs[1]).toHaveTextContent('Release Bundles'); expect(tabs[1]).toHaveClass('active'); // Search bar expect(screen.getByPlaceholderText('Filter by Name')).toBeInTheDocument(); // Date Uploaded can be sorted await userEvent.hover(screen.getByTestId('icon-arrow')); expect(await screen.findByText('Switch to ascending order')).toBeInTheDocument(); await userEvent.click(screen.getByTestId('icon-arrow')); await waitFor(() => { expect(mockRequests.sourceMaps).toHaveBeenLastCalledWith( '/projects/org-slug/project-slug/files/source-maps/', expect.objectContaining({ query: expect.objectContaining({ sortBy: '-date_added', }), }) ); }); // Active tab contains correct link expect(screen.getByRole('link', {name: /Release Bundles/})).toHaveAttribute( 'href', '/settings/org-slug/projects/project-slug/source-maps/release-bundles/' ); // Name expect(await screen.findByRole('link', {name: '1234'})).toBeInTheDocument(); // Artifacts expect(screen.getByText('0')).toBeInTheDocument(); // Date expect(screen.getByText('May 6, 2020 1:41 PM UTC')).toBeInTheDocument(); // Delete buttons (this example renders 2 rows) expect(screen.getAllByRole('button', {name: 'Remove All Artifacts'})).toHaveLength( 2 ); expect( screen.getAllByRole('button', {name: 'Remove All Artifacts'})[0] ).toBeEnabled(); renderGlobalModal(); // Delete item displays a confirmation modal await userEvent.click( screen.getAllByRole('button', {name: 'Remove All Artifacts'})[0] ); expect( await screen.findByText( 'Are you sure you want to remove all artifacts in this archive?' ) ).toBeInTheDocument(); // Close modal await userEvent.click(screen.getByRole('button', {name: 'Cancel'})); // Switch tab await userEvent.click(screen.getByRole('link', {name: 'Artifact Bundles'})); expect(router.push).toHaveBeenCalledWith({ pathname: '/settings/org-slug/projects/project-slug/source-maps/artifact-bundles/', query: undefined, }); }); it('renders empty state', async function () { const {organization, route, project, router, routerContext} = initializeOrg({ router: { location: { query: {}, pathname: `/settings/${initializeOrg().organization.slug}/projects/${ initializeOrg().project.slug }/source-maps/release-bundles/`, }, params: {}, }, }); renderReleaseBundlesMockRequests({ orgSlug: organization.slug, projectSlug: project.slug, empty: true, }); render( , {context: routerContext, organization} ); expect( await screen.findByText('No release bundles found for this project.') ).toBeInTheDocument(); }); }); describe('Artifact Bundles', function () { it('renders default state', async function () { const {organization, route, project, router, routerContext} = initializeOrg({ router: { location: { query: {}, pathname: `/settings/${initializeOrg().organization.slug}/projects/${ initializeOrg().project.slug }/source-maps/artifact-bundles/`, }, params: {}, }, }); const mockRequests = renderDebugIdBundlesMockRequests({ orgSlug: organization.slug, projectSlug: project.slug, }); render( , {context: routerContext, organization} ); // Title expect(screen.getByRole('heading', {name: 'Source Maps'})).toBeInTheDocument(); // Active tab const tabs = screen.getAllByRole('listitem'); expect(tabs).toHaveLength(2); // Tab 1 expect(tabs[0]).toHaveTextContent('Artifact Bundles'); expect(tabs[0]).toHaveClass('active'); // Tab 2 expect(tabs[1]).toHaveTextContent('Release Bundles'); expect(tabs[1]).not.toHaveClass('active'); // Search bar expect(screen.getByPlaceholderText('Filter by Bundle ID')).toBeInTheDocument(); // Date Uploaded can be sorted await userEvent.hover(screen.getByTestId('icon-arrow')); expect(await screen.findByText('Switch to ascending order')).toBeInTheDocument(); await userEvent.click(screen.getByTestId('icon-arrow')); await waitFor(() => { expect(mockRequests.artifactBundles).toHaveBeenLastCalledWith( '/projects/org-slug/project-slug/files/artifact-bundles/', expect.objectContaining({ query: expect.objectContaining({ sortBy: '-date_added', }), }) ); }); // Artifacts expect(screen.getByText('39')).toBeInTheDocument(); // Date Uploaded expect(screen.getByText('Mar 8, 2023 9:53 AM UTC')).toBeInTheDocument(); // Delete button expect(screen.getByRole('button', {name: 'Remove All Artifacts'})).toBeEnabled(); // Click on bundle id await userEvent.click( screen.getByRole('link', {name: 'b916a646-2c6b-4e45-af4c-409830a44e0e'}) ); expect(router.push).toHaveBeenLastCalledWith( '/settings/org-slug/projects/project-slug/source-maps/artifact-bundles/b916a646-2c6b-4e45-af4c-409830a44e0e' ); renderGlobalModal(); // Delete item displays a confirmation modal await userEvent.click(screen.getByRole('button', {name: 'Remove All Artifacts'})); expect( await screen.findByText( 'Are you sure you want to remove all artifacts in this archive?' ) ).toBeInTheDocument(); // Close modal await userEvent.click(screen.getByRole('button', {name: 'Confirm'})); await waitFor(() => { expect(mockRequests.artifactBundlesDeletion).toHaveBeenLastCalledWith( '/projects/org-slug/project-slug/files/artifact-bundles/', expect.objectContaining({ query: expect.objectContaining({ bundleId: 'b916a646-2c6b-4e45-af4c-409830a44e0e', }), }) ); }); // Switch tab await userEvent.click(screen.getByRole('link', {name: /Release Bundles/})); expect(router.push).toHaveBeenCalledWith({ pathname: '/settings/org-slug/projects/project-slug/source-maps/release-bundles/', query: undefined, }); }); it('renders empty state', async function () { const {organization, route, project, router, routerContext} = initializeOrg({ router: { location: { query: {}, pathname: `/settings/${initializeOrg().organization.slug}/projects/${ initializeOrg().project.slug }/source-maps/artifact-bundles/`, }, params: {}, }, }); renderDebugIdBundlesMockRequests({ orgSlug: organization.slug, projectSlug: project.slug, empty: true, }); render( , {context: routerContext, organization} ); expect( await screen.findByText('No debug ID bundles found for this project.') ).toBeInTheDocument(); }); }); });