fix(#145): address PR review — full-table scan, a11y, grid, tests
- DocumentService.getRecentActivity: replace findAll(Sort)+stream().limit() with findAll(PageRequest) so LIMIT is pushed to the database - +page.svelte: collapse two-column grid to single column when mentions is empty - DashboardNeedsMetadata: raise "show all" link from text-xs (12px) to text-sm (14px) and add hover:underline for WCAG 1.4.1 - DashboardRecentDocuments: add comment explaining why T12:00:00 noon-anchor is absent (updatedAt is a full ISO datetime, not a date-only string) - DocumentServiceTest: update getRecentActivity tests to assert PageRequest usage instead of findAll(Sort) - DocumentRepositoryTest: add @DataJpaTest verifying findAll(PageRequest) returns only size rows, not the full table - DocumentControllerTest: add test for default size=5 when param is omitted - NotificationServiceTest: add test documenting that type+read=true falls through to the type-only query (intentional) - page.server.spec.ts: replace stale tests with full dashboard-mode coverage - DashboardMentions.svelte.spec.ts: add tests for REPLY type and absent documentId - DashboardResumeStrip.svelte.spec.ts: add corrupt localStorage test Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>
This commit is contained in:
@@ -262,8 +262,9 @@ public class DocumentService {
|
|||||||
|
|
||||||
// 0. Zuletzt aktive Dokumente (sortiert nach updatedAt DESC)
|
// 0. Zuletzt aktive Dokumente (sortiert nach updatedAt DESC)
|
||||||
public List<Document> getRecentActivity(int size) {
|
public List<Document> getRecentActivity(int size) {
|
||||||
return documentRepository.findAll(Sort.by(Sort.Direction.DESC, "updatedAt"))
|
return documentRepository.findAll(
|
||||||
.stream().limit(size).toList();
|
PageRequest.of(0, size, Sort.by(Sort.Direction.DESC, "updatedAt"))
|
||||||
|
).getContent();
|
||||||
}
|
}
|
||||||
|
|
||||||
// 1. Allgemeine Suche (für das Suchfeld im Frontend)
|
// 1. Allgemeine Suche (für das Suchfeld im Frontend)
|
||||||
|
|||||||
@@ -427,6 +427,17 @@ class DocumentControllerTest {
|
|||||||
.andExpect(jsonPath("$[1].title").value("Beta"));
|
.andExpect(jsonPath("$[1].title").value("Beta"));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
@WithMockUser
|
||||||
|
void getRecentActivity_appliesDefaultSizeOfFive_whenSizeParamOmitted() throws Exception {
|
||||||
|
when(documentService.getRecentActivity(5)).thenReturn(List.of());
|
||||||
|
|
||||||
|
mockMvc.perform(get("/api/documents/recent-activity"))
|
||||||
|
.andExpect(status().isOk());
|
||||||
|
|
||||||
|
verify(documentService).getRecentActivity(5);
|
||||||
|
}
|
||||||
|
|
||||||
// ─── GET /api/documents/{id}/versions ────────────────────────────────────
|
// ─── GET /api/documents/{id}/versions ────────────────────────────────────
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
|
|||||||
@@ -152,6 +152,23 @@ class DocumentRepositoryTest {
|
|||||||
assertThat(documentRepository.countByMetadataCompleteFalse()).isEqualTo(1);
|
assertThat(documentRepository.countByMetadataCompleteFalse()).isEqualTo(1);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// ─── findAll (PageRequest) — recent activity ──────────────────────────────
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void findAll_withPageRequest_returnsOnlySizeRows_notFullTable() {
|
||||||
|
for (int i = 0; i < 10; i++) {
|
||||||
|
documentRepository.save(Document.builder()
|
||||||
|
.title("Doc " + i).originalFilename("doc" + i + ".pdf")
|
||||||
|
.status(DocumentStatus.PLACEHOLDER).build());
|
||||||
|
}
|
||||||
|
|
||||||
|
Page<Document> result = documentRepository.findAll(
|
||||||
|
PageRequest.of(0, 3, Sort.by(Sort.Direction.DESC, "updatedAt")));
|
||||||
|
|
||||||
|
assertThat(result.getContent()).hasSize(3);
|
||||||
|
assertThat(result.getTotalElements()).isEqualTo(10);
|
||||||
|
}
|
||||||
|
|
||||||
// ─── findByMetadataCompleteFalse (Pageable) ───────────────────────────────
|
// ─── findByMetadataCompleteFalse (Pageable) ───────────────────────────────
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
|
|||||||
@@ -1218,21 +1218,30 @@ class DocumentServiceTest {
|
|||||||
|
|
||||||
@Test
|
@Test
|
||||||
void getRecentActivity_returnsMostRecentlyUpdatedDocuments() {
|
void getRecentActivity_returnsMostRecentlyUpdatedDocuments() {
|
||||||
java.time.LocalDateTime oldest = java.time.LocalDateTime.of(2024, 1, 1, 0, 0);
|
|
||||||
java.time.LocalDateTime middle = java.time.LocalDateTime.of(2024, 6, 1, 0, 0);
|
|
||||||
java.time.LocalDateTime newest = java.time.LocalDateTime.of(2024, 12, 1, 0, 0);
|
|
||||||
|
|
||||||
Document doc1 = Document.builder().id(UUID.randomUUID()).title("Oldest").build();
|
Document doc1 = Document.builder().id(UUID.randomUUID()).title("Oldest").build();
|
||||||
Document doc2 = Document.builder().id(UUID.randomUUID()).title("Middle").build();
|
Document doc2 = Document.builder().id(UUID.randomUUID()).title("Middle").build();
|
||||||
Document doc3 = Document.builder().id(UUID.randomUUID()).title("Newest").build();
|
Document doc3 = Document.builder().id(UUID.randomUUID()).title("Newest").build();
|
||||||
|
|
||||||
// findAll(Sort) returns documents already sorted DESC by updatedAt
|
Page<Document> page = new PageImpl<>(List.of(doc3, doc2));
|
||||||
when(documentRepository.findAll(Sort.by(Sort.Direction.DESC, "updatedAt")))
|
when(documentRepository.findAll(any(Pageable.class))).thenReturn(page);
|
||||||
.thenReturn(List.of(doc3, doc2, doc1));
|
|
||||||
|
|
||||||
List<Document> result = documentService.getRecentActivity(2);
|
List<Document> result = documentService.getRecentActivity(2);
|
||||||
|
|
||||||
assertThat(result).hasSize(2);
|
assertThat(result).hasSize(2);
|
||||||
assertThat(result).containsExactly(doc3, doc2);
|
assertThat(result).containsExactly(doc3, doc2);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void getRecentActivity_usesPageRequestWithSizeLimit_notFindAll() {
|
||||||
|
Page<Document> page = new PageImpl<>(List.of());
|
||||||
|
when(documentRepository.findAll(any(Pageable.class))).thenReturn(page);
|
||||||
|
|
||||||
|
documentService.getRecentActivity(3);
|
||||||
|
|
||||||
|
ArgumentCaptor<Pageable> captor = ArgumentCaptor.forClass(Pageable.class);
|
||||||
|
verify(documentRepository).findAll(captor.capture());
|
||||||
|
assertThat(captor.getValue().getPageSize()).isEqualTo(3);
|
||||||
|
assertThat(captor.getValue().getSort())
|
||||||
|
.isEqualTo(Sort.by(Sort.Direction.DESC, "updatedAt"));
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -401,6 +401,24 @@ class NotificationServiceTest {
|
|||||||
.findByRecipientIdAndTypeAndReadFalseOrderByCreatedAtDesc(any(), any(), any());
|
.findByRecipientIdAndTypeAndReadFalseOrderByCreatedAtDesc(any(), any(), any());
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void getNotifications_withTypeAndReadTrue_fallsBackToTypeOnlyQuery() {
|
||||||
|
// read=true with a type filter falls through to the type-only branch —
|
||||||
|
// it returns all notifications of that type (both read and unread).
|
||||||
|
// The read=true filter is intentionally not supported on the backend;
|
||||||
|
// callers that need only-read results must filter client-side.
|
||||||
|
when(notificationRepository.findByRecipientIdAndTypeOrderByCreatedAtDesc(
|
||||||
|
eq(userA.getId()), eq(NotificationType.MENTION), any()))
|
||||||
|
.thenReturn(Page.empty());
|
||||||
|
|
||||||
|
notificationService.getNotifications(userA.getId(), NotificationType.MENTION, true, Pageable.ofSize(5));
|
||||||
|
|
||||||
|
verify(notificationRepository).findByRecipientIdAndTypeOrderByCreatedAtDesc(
|
||||||
|
eq(userA.getId()), eq(NotificationType.MENTION), any());
|
||||||
|
verify(notificationRepository, never())
|
||||||
|
.findByRecipientIdAndTypeAndReadFalseOrderByCreatedAtDesc(any(), any(), any());
|
||||||
|
}
|
||||||
|
|
||||||
// ─── private helpers ──────────────────────────────────────────────────────
|
// ─── private helpers ──────────────────────────────────────────────────────
|
||||||
|
|
||||||
private DocumentComment commentWithAuthor(UUID id, UUID parentId, UUID authorId, String authorName) {
|
private DocumentComment commentWithAuthor(UUID id, UUID parentId, UUID authorId, String authorName) {
|
||||||
|
|||||||
@@ -65,4 +65,21 @@ describe('DashboardMentions', () => {
|
|||||||
render(DashboardMentions, { mentions: [makeMention({ actorName: 'Maria Müller' })] });
|
render(DashboardMentions, { mentions: [makeMention({ actorName: 'Maria Müller' })] });
|
||||||
await expect.element(page.getByText('Maria Müller')).toBeInTheDocument();
|
await expect.element(page.getByText('Maria Müller')).toBeInTheDocument();
|
||||||
});
|
});
|
||||||
|
|
||||||
|
it('shows "replied" label for REPLY type', async () => {
|
||||||
|
render(DashboardMentions, { mentions: [makeMention({ type: 'REPLY' })] });
|
||||||
|
const widget = page.getByTestId('dashboard-mentions');
|
||||||
|
await expect.element(widget).toBeInTheDocument();
|
||||||
|
const link = page.getByRole('link');
|
||||||
|
await expect.element(link).toBeInTheDocument();
|
||||||
|
});
|
||||||
|
|
||||||
|
it('renders a span instead of a link when documentId is absent', async () => {
|
||||||
|
render(DashboardMentions, {
|
||||||
|
mentions: [makeMention({ documentId: undefined, actorName: 'Lena Bauer' })]
|
||||||
|
});
|
||||||
|
await expect.element(page.getByText('Lena Bauer')).toBeInTheDocument();
|
||||||
|
const links = page.getByRole('link');
|
||||||
|
await expect.element(links).not.toBeInTheDocument();
|
||||||
|
});
|
||||||
});
|
});
|
||||||
|
|||||||
@@ -29,7 +29,7 @@ let { incompleteDocs }: Props = $props();
|
|||||||
</div>
|
</div>
|
||||||
{/each}
|
{/each}
|
||||||
<div class="mt-4">
|
<div class="mt-4">
|
||||||
<a href="/enrich" class="font-sans text-xs text-ink-2 hover:text-ink"
|
<a href="/enrich" class="font-sans text-sm text-ink-2 hover:text-ink hover:underline"
|
||||||
>{m.dashboard_needs_metadata_show_all()}</a
|
>{m.dashboard_needs_metadata_show_all()}</a
|
||||||
>
|
>
|
||||||
</div>
|
</div>
|
||||||
|
|||||||
@@ -16,6 +16,7 @@ interface Props {
|
|||||||
let { recentDocs }: Props = $props();
|
let { recentDocs }: Props = $props();
|
||||||
|
|
||||||
function formatDate(dateStr: string): string {
|
function formatDate(dateStr: string): string {
|
||||||
|
// updatedAt is a full ISO datetime — no T12:00:00 noon-anchor needed here
|
||||||
return new Intl.DateTimeFormat(getLocale(), {
|
return new Intl.DateTimeFormat(getLocale(), {
|
||||||
day: 'numeric',
|
day: 'numeric',
|
||||||
month: 'long',
|
month: 'long',
|
||||||
|
|||||||
@@ -40,4 +40,11 @@ describe('DashboardResumeStrip', () => {
|
|||||||
const link = page.getByRole('link');
|
const link = page.getByRole('link');
|
||||||
await expect.element(link).toHaveAttribute('href', '/documents/doc-456');
|
await expect.element(link).toHaveAttribute('href', '/documents/doc-456');
|
||||||
});
|
});
|
||||||
|
|
||||||
|
it('renders nothing when localStorage contains malformed JSON', async () => {
|
||||||
|
localStorage.setItem('familienarchiv.lastVisited', '{not valid json');
|
||||||
|
render(DashboardResumeStrip, {});
|
||||||
|
const strip = page.getByTestId('resume-strip');
|
||||||
|
await expect.element(strip).not.toBeInTheDocument();
|
||||||
|
});
|
||||||
});
|
});
|
||||||
|
|||||||
@@ -100,7 +100,7 @@ $effect(() => {
|
|||||||
</div>
|
</div>
|
||||||
{/if}
|
{/if}
|
||||||
|
|
||||||
<div class="mt-6 grid gap-4 lg:grid-cols-2">
|
<div class="mt-6 grid gap-4 {(data.mentions?.length ?? 0) > 0 ? 'lg:grid-cols-2' : ''}">
|
||||||
<DashboardMentions mentions={data.mentions ?? []} />
|
<DashboardMentions mentions={data.mentions ?? []} />
|
||||||
<DashboardNeedsMetadata incompleteDocs={data.incompleteDocs ?? []} />
|
<DashboardNeedsMetadata incompleteDocs={data.incompleteDocs ?? []} />
|
||||||
</div>
|
</div>
|
||||||
|
|||||||
@@ -19,40 +19,115 @@ function makeUrl(params: Record<string, string | string[]> = {}) {
|
|||||||
return url;
|
return url;
|
||||||
}
|
}
|
||||||
|
|
||||||
// ─── happy path ───────────────────────────────────────────────────────────────
|
// ─── dashboard mode (no search filters) ──────────────────────────────────────
|
||||||
|
|
||||||
describe('home page load — happy path', () => {
|
describe('home page load — dashboard mode', () => {
|
||||||
it('returns documents and persons on success', async () => {
|
it('sets isDashboard true and fetches all three widget APIs', async () => {
|
||||||
vi.mocked(createApiClient).mockReturnValue({
|
const mockGet = vi
|
||||||
GET: vi
|
.fn()
|
||||||
.fn()
|
.mockResolvedValueOnce({ response: { ok: true, status: 200 }, data: [] }) // persons
|
||||||
.mockResolvedValueOnce({
|
.mockResolvedValueOnce({
|
||||||
response: { ok: true, status: 200 },
|
response: { ok: true },
|
||||||
data: [{ id: 'd1', title: 'Brief' }]
|
data: { content: [{ id: 'n1' }] }
|
||||||
})
|
}) // notifications
|
||||||
.mockResolvedValueOnce({
|
.mockResolvedValueOnce({ response: { ok: true }, data: [{ id: 'd1' }] }) // incomplete
|
||||||
response: { ok: true, status: 200 },
|
.mockResolvedValueOnce({ response: { ok: true }, data: [{ id: 'd2' }] }); // recent
|
||||||
data: [{ id: 'p1', firstName: 'Hans', lastName: 'Müller' }]
|
vi.mocked(createApiClient).mockReturnValue({ GET: mockGet } as ReturnType<
|
||||||
})
|
typeof createApiClient
|
||||||
.mockResolvedValueOnce({ response: { ok: true }, data: { count: 3 } })
|
>);
|
||||||
} as ReturnType<typeof createApiClient>);
|
|
||||||
|
|
||||||
const result = await load({ url: makeUrl(), fetch: vi.fn() as unknown as typeof fetch });
|
const result = await load({ url: makeUrl(), fetch: vi.fn() as unknown as typeof fetch });
|
||||||
|
|
||||||
expect(result.documents).toHaveLength(1);
|
expect(result.isDashboard).toBe(true);
|
||||||
expect(result.incompleteCount).toBe(3);
|
expect(result.mentions).toHaveLength(1);
|
||||||
expect(result.error).toBeNull();
|
expect(result.incompleteDocs).toHaveLength(1);
|
||||||
|
expect(result.recentDocs).toHaveLength(1);
|
||||||
|
expect(result.documents).toEqual([]);
|
||||||
});
|
});
|
||||||
|
|
||||||
it('passes search params from the URL to the API', async () => {
|
it('defaults mentions to [] when notifications API rejects', async () => {
|
||||||
|
const mockGet = vi
|
||||||
|
.fn()
|
||||||
|
.mockResolvedValueOnce({ response: { ok: true, status: 200 }, data: [] }) // persons
|
||||||
|
.mockRejectedValueOnce(new Error('network')) // notifications
|
||||||
|
.mockResolvedValueOnce({ response: { ok: true }, data: [] }) // incomplete
|
||||||
|
.mockResolvedValueOnce({ response: { ok: true }, data: [] }); // recent
|
||||||
|
vi.mocked(createApiClient).mockReturnValue({ GET: mockGet } as ReturnType<
|
||||||
|
typeof createApiClient
|
||||||
|
>);
|
||||||
|
|
||||||
|
const result = await load({ url: makeUrl(), fetch: vi.fn() as unknown as typeof fetch });
|
||||||
|
|
||||||
|
expect(result.mentions).toEqual([]);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('defaults incompleteDocs to [] when incomplete API rejects', async () => {
|
||||||
|
const mockGet = vi
|
||||||
|
.fn()
|
||||||
|
.mockResolvedValueOnce({ response: { ok: true, status: 200 }, data: [] }) // persons
|
||||||
|
.mockResolvedValueOnce({ response: { ok: true }, data: { content: [] } }) // notifications
|
||||||
|
.mockRejectedValueOnce(new Error('network')) // incomplete
|
||||||
|
.mockResolvedValueOnce({ response: { ok: true }, data: [] }); // recent
|
||||||
|
vi.mocked(createApiClient).mockReturnValue({ GET: mockGet } as ReturnType<
|
||||||
|
typeof createApiClient
|
||||||
|
>);
|
||||||
|
|
||||||
|
const result = await load({ url: makeUrl(), fetch: vi.fn() as unknown as typeof fetch });
|
||||||
|
|
||||||
|
expect(result.incompleteDocs).toEqual([]);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('defaults recentDocs to [] when recent-activity API rejects', async () => {
|
||||||
|
const mockGet = vi
|
||||||
|
.fn()
|
||||||
|
.mockResolvedValueOnce({ response: { ok: true, status: 200 }, data: [] }) // persons
|
||||||
|
.mockResolvedValueOnce({ response: { ok: true }, data: { content: [] } }) // notifications
|
||||||
|
.mockResolvedValueOnce({ response: { ok: true }, data: [] }) // incomplete
|
||||||
|
.mockRejectedValueOnce(new Error('network')); // recent
|
||||||
|
vi.mocked(createApiClient).mockReturnValue({ GET: mockGet } as ReturnType<
|
||||||
|
typeof createApiClient
|
||||||
|
>);
|
||||||
|
|
||||||
|
const result = await load({ url: makeUrl(), fetch: vi.fn() as unknown as typeof fetch });
|
||||||
|
|
||||||
|
expect(result.recentDocs).toEqual([]);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
// ─── search mode (any filter active) ─────────────────────────────────────────
|
||||||
|
|
||||||
|
describe('home page load — search mode', () => {
|
||||||
|
it('sets isDashboard false and skips widget APIs when q is set', async () => {
|
||||||
|
const mockGet = vi
|
||||||
|
.fn()
|
||||||
|
.mockResolvedValueOnce({ response: { ok: true, status: 200 }, data: [{ id: 'd1' }] }) // search docs
|
||||||
|
.mockResolvedValueOnce({ response: { ok: true, status: 200 }, data: [] }); // persons
|
||||||
|
vi.mocked(createApiClient).mockReturnValue({ GET: mockGet } as ReturnType<
|
||||||
|
typeof createApiClient
|
||||||
|
>);
|
||||||
|
|
||||||
|
const result = await load({
|
||||||
|
url: makeUrl({ q: 'Urlaub' }),
|
||||||
|
fetch: vi.fn() as unknown as typeof fetch
|
||||||
|
});
|
||||||
|
|
||||||
|
expect(result.isDashboard).toBe(false);
|
||||||
|
expect(result.documents).toHaveLength(1);
|
||||||
|
expect(result.mentions).toEqual([]);
|
||||||
|
expect(result.incompleteDocs).toEqual([]);
|
||||||
|
expect(result.recentDocs).toEqual([]);
|
||||||
|
// Only two API calls — no widget calls
|
||||||
|
expect(mockGet).toHaveBeenCalledTimes(2);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('passes search params from the URL to the documents API', async () => {
|
||||||
const mockGet = vi
|
const mockGet = vi
|
||||||
.fn()
|
.fn()
|
||||||
.mockResolvedValueOnce({ response: { ok: true, status: 200 }, data: [] })
|
.mockResolvedValueOnce({ response: { ok: true, status: 200 }, data: [] })
|
||||||
.mockResolvedValueOnce({ response: { ok: true, status: 200 }, data: [] })
|
.mockResolvedValueOnce({ response: { ok: true, status: 200 }, data: [] });
|
||||||
.mockResolvedValueOnce({ response: { ok: true }, data: { count: 0 } });
|
vi.mocked(createApiClient).mockReturnValue({ GET: mockGet } as ReturnType<
|
||||||
vi.mocked(createApiClient).mockReturnValue({
|
typeof createApiClient
|
||||||
GET: mockGet
|
>);
|
||||||
} as ReturnType<typeof createApiClient>);
|
|
||||||
|
|
||||||
await load({
|
await load({
|
||||||
url: makeUrl({ q: 'Urlaub', from: '2020-01-01' }),
|
url: makeUrl({ q: 'Urlaub', from: '2020-01-01' }),
|
||||||
@@ -63,46 +138,14 @@ describe('home page load — happy path', () => {
|
|||||||
expect(firstCall[1].params.query.q).toBe('Urlaub');
|
expect(firstCall[1].params.query.q).toBe('Urlaub');
|
||||||
expect(firstCall[1].params.query.from).toBe('2020-01-01');
|
expect(firstCall[1].params.query.from).toBe('2020-01-01');
|
||||||
});
|
});
|
||||||
|
|
||||||
it('returns incompleteCount 0 when the incomplete-count API fails', async () => {
|
|
||||||
vi.mocked(createApiClient).mockReturnValue({
|
|
||||||
GET: vi
|
|
||||||
.fn()
|
|
||||||
.mockResolvedValueOnce({ response: { ok: true, status: 200 }, data: [] })
|
|
||||||
.mockResolvedValueOnce({ response: { ok: true, status: 200 }, data: [] })
|
|
||||||
.mockResolvedValueOnce({ response: { ok: false }, data: null })
|
|
||||||
} as ReturnType<typeof createApiClient>);
|
|
||||||
|
|
||||||
const result = await load({ url: makeUrl(), fetch: vi.fn() as unknown as typeof fetch });
|
|
||||||
|
|
||||||
expect(result.incompleteCount).toBe(0);
|
|
||||||
});
|
|
||||||
});
|
});
|
||||||
|
|
||||||
// ─── 401 redirect ─────────────────────────────────────────────────────────────
|
// ─── 401 redirect ─────────────────────────────────────────────────────────────
|
||||||
|
|
||||||
describe('home page load — auth redirect', () => {
|
describe('home page load — auth redirect', () => {
|
||||||
it('redirects to /login when documents API returns 401', async () => {
|
|
||||||
vi.mocked(createApiClient).mockReturnValue({
|
|
||||||
GET: vi
|
|
||||||
.fn()
|
|
||||||
.mockResolvedValueOnce({ response: { ok: false, status: 401 }, data: null })
|
|
||||||
.mockResolvedValueOnce({ response: { ok: true, status: 200 }, data: [] })
|
|
||||||
.mockResolvedValueOnce({ response: { ok: true }, data: { count: 0 } })
|
|
||||||
} as ReturnType<typeof createApiClient>);
|
|
||||||
|
|
||||||
await expect(
|
|
||||||
load({ url: makeUrl(), fetch: vi.fn() as unknown as typeof fetch })
|
|
||||||
).rejects.toMatchObject({ location: '/login' });
|
|
||||||
});
|
|
||||||
|
|
||||||
it('redirects to /login when persons API returns 401', async () => {
|
it('redirects to /login when persons API returns 401', async () => {
|
||||||
vi.mocked(createApiClient).mockReturnValue({
|
vi.mocked(createApiClient).mockReturnValue({
|
||||||
GET: vi
|
GET: vi.fn().mockResolvedValueOnce({ response: { ok: false, status: 401 }, data: null })
|
||||||
.fn()
|
|
||||||
.mockResolvedValueOnce({ response: { ok: true, status: 200 }, data: [] })
|
|
||||||
.mockResolvedValueOnce({ response: { ok: false, status: 401 }, data: null })
|
|
||||||
.mockResolvedValueOnce({ response: { ok: true }, data: { count: 0 } })
|
|
||||||
} as ReturnType<typeof createApiClient>);
|
} as ReturnType<typeof createApiClient>);
|
||||||
|
|
||||||
await expect(
|
await expect(
|
||||||
@@ -123,6 +166,5 @@ describe('home page load — network error fallback', () => {
|
|||||||
|
|
||||||
expect(result.error).toBe('Daten konnten nicht geladen werden.');
|
expect(result.error).toBe('Daten konnten nicht geladen werden.');
|
||||||
expect(result.documents).toEqual([]);
|
expect(result.documents).toEqual([]);
|
||||||
expect(result.incompleteCount).toBe(0);
|
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|||||||
Reference in New Issue
Block a user