123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178 |
- import {EventFixture} from 'sentry-fixture/event';
- import {GroupFixture} from 'sentry-fixture/group';
- import {OrganizationFixture} from 'sentry-fixture/organization';
- import {ProjectFixture} from 'sentry-fixture/project';
- import {render, screen, userEvent, waitFor} from 'sentry-test/reactTestingLibrary';
- import {getConfigForIssueType} from 'sentry/utils/issueTypeConfig';
- import SolutionsSection from 'sentry/views/issueDetails/streamline/solutionsSection';
- jest.mock('sentry/utils/issueTypeConfig');
- describe('SolutionsSection', () => {
- const mockEvent = EventFixture();
- const mockGroup = GroupFixture();
- const mockProject = ProjectFixture();
- const organization = OrganizationFixture({genAIConsent: true, hideAiFeatures: false});
- beforeEach(() => {
- MockApiClient.clearMockResponses();
- MockApiClient.addMockResponse({
- url: `/issues/${mockGroup.id}/autofix/setup/`,
- body: {
- genAIConsent: {ok: true},
- integration: {ok: true},
- githubWriteIntegration: {ok: true},
- },
- });
- jest.mocked(getConfigForIssueType).mockReturnValue({
- issueSummary: {
- enabled: true,
- },
- resources: {
- description: 'Test Resource',
- links: [{link: 'https://example.com', text: 'Test Link'}],
- linksByPlatform: {},
- },
- actions: {
- archiveUntilOccurrence: {enabled: false},
- delete: {enabled: false},
- deleteAndDiscard: {enabled: false},
- ignore: {enabled: false},
- merge: {enabled: false},
- resolveInRelease: {enabled: false},
- share: {enabled: false},
- },
- aiSuggestedSolution: false,
- attachments: {enabled: false},
- autofix: true,
- discover: {enabled: false},
- events: {enabled: false},
- evidence: null,
- filterAndSearchHeader: {enabled: false},
- mergedIssues: {enabled: false},
- performanceDurationRegression: {enabled: false},
- profilingDurationRegression: {enabled: false},
- regression: {enabled: false},
- replays: {enabled: false},
- showFeedbackWidget: false,
- similarIssues: {enabled: false},
- spanEvidence: {enabled: false},
- stacktrace: {enabled: false},
- stats: {enabled: false},
- tags: {enabled: false},
- tagsTab: {enabled: false},
- userFeedback: {enabled: false},
- usesIssuePlatform: false,
- });
- });
- it('renders loading state when summary is pending', () => {
- // Use a delayed response to simulate loading state
- MockApiClient.addMockResponse({
- url: `/organizations/${mockProject.organization.slug}/issues/${mockGroup.id}/summarize/`,
- method: 'POST',
- statusCode: 200,
- body: new Promise(() => {}), // Never resolves, keeping the loading state
- });
- render(
- <SolutionsSection event={mockEvent} group={mockGroup} project={mockProject} />,
- {
- organization,
- }
- );
- expect(screen.getByText('Solutions Hub')).toBeInTheDocument();
- expect(screen.getAllByTestId('loading-placeholder')).toHaveLength(3);
- });
- it('renders summary when AI features are enabled and data is available', async () => {
- const mockSummary = 'This is a test summary';
- MockApiClient.addMockResponse({
- url: `/organizations/${mockProject.organization.slug}/issues/${mockGroup.id}/summarize/`,
- method: 'POST',
- body: {
- whatsWrong: mockSummary,
- },
- });
- render(
- <SolutionsSection event={mockEvent} group={mockGroup} project={mockProject} />,
- {
- organization,
- }
- );
- await waitFor(() => {
- expect(screen.getByText(mockSummary)).toBeInTheDocument();
- expect(
- screen.getByRole('button', {name: 'Open Solutions Hub'})
- ).toBeInTheDocument();
- });
- });
- it('renders AI setup prompt when consent is not given', () => {
- const customOrganization = OrganizationFixture({
- genAIConsent: false,
- hideAiFeatures: false,
- });
- render(
- <SolutionsSection event={mockEvent} group={mockGroup} project={mockProject} />,
- {
- organization: customOrganization,
- }
- );
- expect(
- screen.getByText('Explore potential root causes and solutions with Sentry AI.')
- ).toBeInTheDocument();
- expect(screen.getByRole('button', {name: 'Open Solutions Hub'})).toBeInTheDocument();
- });
- it('renders resources section when AI features are disabled', () => {
- const customOrganization = OrganizationFixture({
- hideAiFeatures: true,
- genAIConsent: false,
- });
- render(
- <SolutionsSection event={mockEvent} group={mockGroup} project={mockProject} />,
- {
- organization: customOrganization,
- }
- );
- expect(screen.getByText('Test Link')).toBeInTheDocument();
- expect(screen.getByRole('button', {name: 'READ MORE'})).toBeInTheDocument();
- });
- it('toggles resources content when clicking Read More/Show Less', async () => {
- const customOrganization = OrganizationFixture({
- hideAiFeatures: true,
- genAIConsent: false,
- });
- render(
- <SolutionsSection event={mockEvent} group={mockGroup} project={mockProject} />,
- {
- organization: customOrganization,
- }
- );
- const readMoreButton = screen.getByRole('button', {name: 'READ MORE'});
- await userEvent.click(readMoreButton);
- expect(screen.getByRole('button', {name: 'SHOW LESS'})).toBeInTheDocument();
- const showLessButton = screen.getByRole('button', {name: 'SHOW LESS'});
- await userEvent.click(showLessButton);
- expect(screen.queryByRole('button', {name: 'SHOW LESS'})).not.toBeInTheDocument();
- expect(screen.getByRole('button', {name: 'READ MORE'})).toBeInTheDocument();
- });
- });
|