feat: add multi-provider support (Komga + Stripstream Librarian)
Some checks failed
Deploy with Docker Compose / deploy (push) Has been cancelled

- Introduce provider abstraction layer (IMediaProvider, KomgaProvider, StripstreamProvider)
- Add Stripstream Librarian as second media provider with full feature parity
- Migrate all pages and components from direct Komga services to provider factory
- Remove dead service code (BaseApiService, HomeService, LibraryService, SearchService, TestService)
- Fix library/series page-based pagination for both providers (Komga 0-indexed, Stripstream 1-indexed)
- Fix unread filter and search on library page for both providers
- Fix read progress display for Stripstream (reading_status mapping)
- Fix series read status (books_read_count) for Stripstream
- Add global search with series results for Stripstream (series_hits from Meilisearch)
- Fix thumbnail proxy to return 404 gracefully instead of JSON on upstream error
- Replace duration-based cache debug detection with x-nextjs-cache header

Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>
This commit is contained in:
2026-03-11 11:48:17 +01:00
parent a1a95775db
commit 7d0f1c4457
77 changed files with 2695 additions and 1705 deletions

View File

@@ -5,7 +5,7 @@ import { SeriesList } from "./SeriesList";
import { Pagination } from "@/components/ui/Pagination";
import { useRouter, usePathname, useSearchParams } from "next/navigation";
import { useState, useEffect, useCallback } from "react";
import type { KomgaSeries } from "@/types/komga";
import type { NormalizedSeries } from "@/lib/providers/types";
import { SearchInput } from "./SearchInput";
import { useTranslate } from "@/hooks/useTranslate";
import { PageSizeSelect } from "@/components/common/PageSizeSelect";
@@ -15,7 +15,7 @@ import { UnreadFilterButton } from "@/components/common/UnreadFilterButton";
import { updatePreferences as updatePreferencesAction } from "@/app/actions/preferences";
interface PaginatedSeriesGridProps {
series: KomgaSeries[];
series: NormalizedSeries[];
currentPage: number;
totalPages: number;
totalElements: number;
@@ -108,19 +108,13 @@ export function PaginatedSeriesGrid({
const handleUnreadFilter = async () => {
const newUnreadState = !showOnlyUnread;
setShowOnlyUnread(newUnreadState);
await updateUrlParams({
page: "1",
unread: newUnreadState ? "true" : "false",
});
await updateUrlParams({ page: "1", unread: newUnreadState ? "true" : "false" });
await persistPreferences({ showOnlyUnread: newUnreadState });
};
const handlePageSizeChange = async (size: number) => {
setCurrentPageSize(size);
await updateUrlParams({
page: "1",
size: size.toString(),
});
await updateUrlParams({ page: "1", size: size.toString() });
await persistPreferences({
displayMode: {