import {AutofixDataFixture} from 'sentry-fixture/autofixData'; import {AutofixStepFixture} from 'sentry-fixture/autofixStep'; import {EventFixture} from 'sentry-fixture/event'; import {FrameFixture} from 'sentry-fixture/frame'; import {GroupFixture} from 'sentry-fixture/group'; import {OrganizationFixture} from 'sentry-fixture/organization'; import {ProjectFixture} from 'sentry-fixture/project'; import { render, screen, userEvent, waitFor, waitForElementToBeRemoved, } from 'sentry-test/reactTestingLibrary'; import {t} from 'sentry/locale'; import {EntryType} from 'sentry/types/event'; import {SolutionsHubDrawer} from 'sentry/views/issueDetails/streamline/sidebar/solutionsHubDrawer'; describe('SolutionsHubDrawer', () => { const organization = OrganizationFixture({ genAIConsent: true, hideAiFeatures: false, features: ['gen-ai-features'], }); const mockEvent = EventFixture({ entries: [ { type: EntryType.EXCEPTION, data: {values: [{stacktrace: {frames: [FrameFixture()]}}]}, }, ], }); const mockGroup = GroupFixture(); const mockProject = ProjectFixture(); const mockAutofixData = AutofixDataFixture({steps: [AutofixStepFixture()]}); beforeEach(() => { MockApiClient.clearMockResponses(); MockApiClient.addMockResponse({ url: `/issues/${mockGroup.id}/autofix/setup/`, body: { genAIConsent: {ok: true}, integration: {ok: true}, githubWriteIntegration: {ok: true}, }, }); MockApiClient.addMockResponse({ url: `/organizations/${mockProject.organization.slug}/issues/${mockGroup.id}/summarize/`, method: 'POST', body: { whatsWrong: 'Test whats wrong', trace: 'Test trace', possibleCause: 'Test possible cause', headline: 'Test headline', }, }); }); it('renders consent state if not consented', async () => { MockApiClient.addMockResponse({ url: `/issues/${mockGroup.id}/autofix/setup/`, body: { genAIConsent: {ok: false}, integration: {ok: false}, githubWriteIntegration: {ok: false}, }, }); MockApiClient.addMockResponse({ url: `/issues/${mockGroup.id}/autofix/`, body: {autofix: null}, }); render( , {organization} ); expect(screen.getByTestId('ai-setup-loading-indicator')).toBeInTheDocument(); await waitForElementToBeRemoved(() => screen.queryByTestId('ai-setup-loading-indicator') ); expect(screen.getByText(mockEvent.id)).toBeInTheDocument(); expect(screen.getByRole('heading', {name: 'Solutions Hub'})).toBeInTheDocument(); expect(screen.getByTestId('ai-setup-data-consent')).toBeInTheDocument(); }); it('renders initial state correctly', async () => { MockApiClient.addMockResponse({ url: `/issues/${mockGroup.id}/autofix/`, body: {autofix: null}, }); render( , {organization} ); expect(screen.getByTestId('ai-setup-loading-indicator')).toBeInTheDocument(); await waitForElementToBeRemoved(() => screen.queryByTestId('ai-setup-loading-indicator') ); expect(screen.getByRole('heading', {name: 'Solutions Hub'})).toBeInTheDocument(); const startButton = screen.getByRole('button', {name: 'Start Autofix'}); expect(startButton).toBeInTheDocument(); }); it('triggers autofix on clicking the Start button', async () => { MockApiClient.addMockResponse({ url: `/issues/${mockGroup.id}/autofix/`, method: 'POST', body: {autofix: null}, }); MockApiClient.addMockResponse({ url: `/issues/${mockGroup.id}/autofix/`, method: 'GET', body: {autofix: null}, }); render( , {organization} ); expect(screen.getByTestId('ai-setup-loading-indicator')).toBeInTheDocument(); await waitForElementToBeRemoved(() => screen.queryByTestId('ai-setup-loading-indicator') ); const startButton = screen.getByRole('button', {name: 'Start Autofix'}); await userEvent.click(startButton); expect( await screen.findByRole('button', {name: t('Start Over')}) ).toBeInTheDocument(); }); it('displays autofix steps and Start Over button when autofixData is available', async () => { MockApiClient.addMockResponse({ url: `/issues/${mockGroup.id}/autofix/`, body: {autofix: mockAutofixData}, }); render( , {organization} ); expect( await screen.findByRole('button', {name: t('Start Over')}) ).toBeInTheDocument(); }); it('resets autofix on clicking the start over button', async () => { MockApiClient.addMockResponse({ url: `/issues/${mockGroup.id}/autofix/`, body: {autofix: mockAutofixData}, }); render( , {organization} ); const startOverButton = await screen.findByRole('button', {name: t('Start Over')}); expect(startOverButton).toBeInTheDocument(); await userEvent.click(startOverButton); await waitFor(() => { expect(screen.getByRole('button', {name: 'Start Autofix'})).toBeInTheDocument(); }); }); it('shows setup if not complete', async () => { MockApiClient.addMockResponse({ url: `/issues/${mockGroup.id}/autofix/setup/`, body: { genAIConsent: {ok: true}, integration: {ok: false}, githubWriteIntegration: {ok: false, repos: []}, }, }); MockApiClient.addMockResponse({ url: `/issues/${mockGroup.id}/autofix/`, body: {autofix: null}, }); MockApiClient.addMockResponse({ url: '/organizations/org-slug/integrations/?provider_key=github&includeConfig=0', body: [], }); render( , {organization} ); expect(screen.getByTestId('ai-setup-loading-indicator')).toBeInTheDocument(); await waitForElementToBeRemoved(() => screen.queryByTestId('ai-setup-loading-indicator') ); expect(screen.getByRole('heading', {name: 'Solutions Hub'})).toBeInTheDocument(); expect(screen.queryByRole('button', {name: 'Start Autofix'})).not.toBeInTheDocument(); expect(await screen.findByText('Install the GitHub Integration')).toBeInTheDocument(); }); });