Skip to content
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
Original file line number Diff line number Diff line change
@@ -0,0 +1,234 @@
import type { JSX } from 'solid-js';

import { MetaProvider } from '@solidjs/meta';
import {
Route,
Router,
MemoryRouter,
createMemoryHistory,
} from '@solidjs/router';
import { render, renderHook, screen, waitFor } from '@solidjs/testing-library';
import { QueryClient, QueryClientProvider } from '@tanstack/solid-query';
import { HttpResponse, http } from 'msw';
import toast from 'solid-toast';
import { describe, expect, test, vi } from 'vitest';

import App from '@/App';
import PostContent from '@/components/modules/PostContent/PostContent';
import useFetchPost from '@/hooks/useFetchPost';
import { server } from '@/mocks/server';

const apiEndpointUrl = import.meta.env.VITE_PUBLIC_API_URL;

vi.mock('solid-toast', () => ({
default: {
error: vi.fn(),
},
Toaster: vi.fn(),
}));

describe('PostContent component', () => {
test('should render "Loading..." element before Data Fetching', async () => {
const queryClient = new QueryClient();

render(() => <PostContent />, {
wrapper: props => (
<MetaProvider>
<QueryClientProvider client={queryClient}>
<Router
// eslint-disable-next-line @typescript-eslint/no-shadow
root={props => <>{props.children}</>}
>
<Route path="/" component={() => <>{props.children}</>} />
</Router>
</QueryClientProvider>
</MetaProvider>
),
});

// @ https://github.com/testing-library/react-hooks-testing-library/issues/23#issuecomment-477542354
const wrapper = (props: { children: JSX.Element }) => (
<QueryClientProvider client={queryClient}>
<Router
// eslint-disable-next-line @typescript-eslint/no-shadow
root={props => <>{props.children}</>}
>
<Route path="/" component={() => <>{props.children}</>} />
</Router>
</QueryClientProvider>
);

renderHook(() => useFetchPost('1'), { wrapper });

// screen.getByRole('button', { name: '' });
const loadingEle = screen.getByText('Loading...'); // substring match
expect(loadingEle).toBeInTheDocument();
});

test('should success to fetch data', async () => {
const queryClient = new QueryClient();

render(() => <PostContent />, {
wrapper: props => (
<MetaProvider>
<QueryClientProvider client={queryClient}>
<Router
// eslint-disable-next-line @typescript-eslint/no-shadow
root={props => <>{props.children}</>}
>
<Route path="/" component={() => <>{props.children}</>} />
</Router>
</QueryClientProvider>
</MetaProvider>
),
});

// expect(await screen.findByText(/qui est esse/)).toBeInTheDocument();

// @ https://github.com/testing-library/react-hooks-testing-library/issues/23#issuecomment-477542354
const wrapper = (props: { children: JSX.Element }) => (
<QueryClientProvider client={queryClient}>
<Router
// eslint-disable-next-line @typescript-eslint/no-shadow
root={props => <>{props.children}</>}
>
<Route path="/" component={() => <>{props.children}</>} />
</Router>
</QueryClientProvider>
);

const { result } = renderHook(() => useFetchPost('1'), { wrapper });

// screen.getByRole('');

await waitFor(() => {
expect(result.isSuccess).toBe(true);
});

await waitFor(() => {
expect(result.data?.id).toBe(1);
});

await waitFor(async () => {
expect(result.data?.title).toEqual(
'sunt aut facere repellat provident occaecati excepturi optio reprehenderit',
);
});
});

test('should render post title and body', async () => {
// Mock window.ResizeObserver @ https://vitest.dev/guide/mocking.html#globals
const ResizeObserverMock = vi.fn(() => ({
disconnect: vi.fn(),
observe: vi.fn(),
takeRecords: vi.fn(),
unobserve: vi.fn(),
}));
vi.stubGlobal('ResizeObserver', ResizeObserverMock);
// now you can access it as `ResizeObserver` or `window.ResizeObserver`

const queryClient = new QueryClient();

// @see https://github.com/solidjs/solid-router/issues/335#issuecomment-1918115042
const history = createMemoryHistory();
history.set({ value: '/post/1' });

render(() => (
<MetaProvider>
<QueryClientProvider client={queryClient}>
<MemoryRouter root={App} history={history}>
<Route path="/post/:id" component={PostContent} />
</MemoryRouter>
</QueryClientProvider>
</MetaProvider>
));

await waitFor(async () => {
expect(
screen.getByText(
'sunt aut facere repellat provident occaecati excepturi optio reprehenderit',
),
).toBeInTheDocument();
});

await waitFor(async () => {
expect(
screen.getByText(
'quia et suscipit suscipit recusandae consequuntur expedita et cum reprehenderit molestiae ut ut quas totam nostrum rerum est autem sunt rem eveniet architecto',
),
).toBeInTheDocument();
});
});

test('should handle API call failure', async () => {
const errorMessage = 'Failed to fetch data.';
const toastErrorSpy = vi.spyOn(toast, 'error');

server.use(
http.get(
`${apiEndpointUrl}/posts/1`,
() =>
new HttpResponse(null, {
status: 500,
}),
),
);

const queryClient = new QueryClient({
defaultOptions: { queries: { retry: false } }, // @ https://github.com/TanStack/query/discussions/2300#discussioncomment-811768
});

render(() => <PostContent />, {
wrapper: props => (
<MetaProvider>
<QueryClientProvider client={queryClient}>
<Router
// eslint-disable-next-line @typescript-eslint/no-shadow
root={props => <>{props.children}</>}
>
<Route path="/" component={() => <>{props.children}</>} />
</Router>
</QueryClientProvider>
</MetaProvider>
),
});

const wrapper = (props: { children: JSX.Element }) => (
<QueryClientProvider client={queryClient}>
<Router
// eslint-disable-next-line @typescript-eslint/no-shadow
root={props => <>{props.children}</>}
>
<Route
path="/"
component={() => (
<>
{props.children}
{/* <Toaster position="bottom-center" /> */}
</>
)}
/>
</Router>
</QueryClientProvider>
);

// screen.getByRole('');
const { result } = renderHook(() => useFetchPost('1'), {
wrapper,
});

await waitFor(() => expect(result.isError).toBe(true));

await waitFor(() => {
expect(result.error).toEqual(Error('Failed to fetch data'));
});

await waitFor(() => {
expect(toastErrorSpy).toHaveBeenCalledWith(errorMessage);
});

// expect(
// await screen.findByText('Failed to fetch data.'),
// ).toBeInTheDocument();
});
});
1 change: 0 additions & 1 deletion apps/solidjs-boilerplate/src/hooks/useFetchPost.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -27,7 +27,6 @@ const useFetchPost = (id: string) => {
if (!result.ok) throw new Error('Failed to fetch data');
return result.json();
},

staleTime: 1000 * 60 * 5, // 5 minutes
throwOnError: true, // Throw an error if the query fails
}));
Expand Down
3 changes: 3 additions & 0 deletions apps/solidjs-boilerplate/src/mocks/handlers.ts
Original file line number Diff line number Diff line change
Expand Up @@ -18,4 +18,7 @@ export const handlers = [
http.get(`${apiEndpointUrl}/posts`, () => {
return HttpResponse.json(posts);
}),
http.get(`${apiEndpointUrl}/posts/1`, () => {
return HttpResponse.json(posts[0]);
}),
];