import * as React from 'react';
import {createMemoryHistory, Route, Router, RouterContext} from 'react-router';
import {render, screen, waitFor} from 'sentry-test/reactTestingLibrary';
import useApiRequests from 'sentry/utils/useApiRequests';
import {RouteContext} from 'sentry/views/routeContext';
describe('useApiRequests', () => {
afterEach(() => {
MockApiClient.clearMockResponses();
});
describe('error handling', () => {
function HomePage() {
const {renderComponent, data} = useApiRequests<{message: {value?: string}}>({
endpoints: [['message', '/some/path/to/something/']],
shouldRenderBadRequests: true,
});
return renderComponent(
{data.message?.value}
);
}
function UniqueErrorsAsyncComponent() {
const {renderComponent, data} = useApiRequests({
endpoints: [
['first', '/first/path/'],
['second', '/second/path/'],
['third', '/third/path/'],
],
shouldRenderBadRequests: true,
});
// @ts-expect-error
return renderComponent({data.message?.value}
);
}
const memoryHistory = createMemoryHistory();
memoryHistory.push('/');
function App() {
return (
{
return (
);
}}
>
);
}
it('renders on successful request', async function () {
const mockRequest = MockApiClient.addMockResponse({
url: '/some/path/to/something/',
method: 'GET',
body: {
value: 'hi',
},
});
render();
await waitFor(() => {
expect(screen.getByText('hi')).toBeInTheDocument();
});
expect(mockRequest).toHaveBeenCalledTimes(1);
});
it('renders error message', async function () {
MockApiClient.addMockResponse({
url: '/some/path/to/something/',
method: 'GET',
body: {
detail: 'oops there was a problem',
},
statusCode: 400,
});
render();
await waitFor(() => {
expect(screen.getByText('oops there was a problem')).toBeInTheDocument();
});
});
it('renders only unique error message', async function () {
MockApiClient.addMockResponse({
url: '/first/path/',
method: 'GET',
body: {
detail: 'oops there was a problem 1',
},
statusCode: 400,
});
MockApiClient.addMockResponse({
url: '/second/path/',
method: 'GET',
body: {
detail: 'oops there was a problem 2',
},
statusCode: 400,
});
MockApiClient.addMockResponse({
url: '/third/path/',
method: 'GET',
body: {
detail: 'oops there was a different problem',
},
statusCode: 400,
});
memoryHistory.push('/unique-error');
render();
await waitFor(() => {
expect(screen.getByTestId('loading-error')).toHaveTextContent(
'oops there was a problem 1'
);
});
await waitFor(() => {
expect(screen.getByTestId('loading-error')).toHaveTextContent(
'oops there was a problem 2'
);
});
await waitFor(() => {
expect(screen.getByTestId('loading-error')).toHaveTextContent(
'oops there was a different problem'
);
});
});
});
describe('slow requests', () => {
it('calls onLoadAllEndpointsSuccess when all endpoints have been loaded', async () => {
const requestQueue: Array<() => void> = [];
jest.spyOn(MockApiClient.prototype, 'request').mockImplementation((_, options) => {
requestQueue.push(() => {
return options?.success?.({message: 'good'});
});
});
const mockOnAllEndpointsSuccess = jest.fn();
function MultiRouteComponent() {
const {remainingRequests} = useApiRequests({
endpoints: [
['data', '/some/path/to/something/'],
['project', '/another/path/here'],
],
onLoadAllEndpointsSuccess: mockOnAllEndpointsSuccess,
});
return {remainingRequests}
;
}
const memoryHistory = createMemoryHistory();
memoryHistory.push('/multi');
function App() {
return (
{
return (
);
}}
>
);
}
render();
expect(await screen.findByText('2')).toBeInTheDocument();
const req1 = requestQueue.shift();
req1?.();
expect(await screen.findByText('1')).toBeInTheDocument();
const req2 = requestQueue.shift();
req2?.();
expect(await screen.findByText('0')).toBeInTheDocument();
expect(mockOnAllEndpointsSuccess).toHaveBeenCalled();
jest.restoreAllMocks();
});
});
});