mirror of
https://github.com/kikootwo/ReadMeABook.git
synced 2026-06-03 12:50:09 +00:00
Compare commits
73 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
| c35bec9f89 | |||
| 09e1a0db3a | |||
| 832a8ad00b | |||
| cc8e106a2b | |||
| 079a337f1c | |||
| 6025ac200a | |||
| 248bd5359c | |||
| 53c1e0dad7 | |||
| 45c8b614e3 | |||
| 24aa6afefc | |||
| 81813dc625 | |||
| f65cb59a9c | |||
| d1ea65a41a | |||
| a5e7af1a53 | |||
| ca02b8b6e7 | |||
| 85aa80938a | |||
| efb4f64014 | |||
| 95917715b1 | |||
| a50fbc721e | |||
| d6eca611fc | |||
| 45e818c181 | |||
| 85977d123c | |||
| 441724c378 | |||
| d0ce485bdc | |||
| c29cfa3a07 | |||
| 7f706e806f | |||
| 338331d006 | |||
| 6ca2e964e8 | |||
| 1d1aaa7ff3 | |||
| cbf02d3e24 | |||
| f0b2476b87 | |||
| 04b6a2c135 | |||
| 6da2c4ce95 | |||
| ce8f4d642b | |||
| 61b183542c | |||
| ae4a73144d | |||
| c57d0c1492 | |||
| 8f8387abff | |||
| 4ae68d01de | |||
| 225ef8c919 | |||
| 610873af6b | |||
| ff80d995c5 | |||
| e4e127880b | |||
| bfd624e120 | |||
| b559835390 | |||
| d25a6ebf79 | |||
| b3dad47aba | |||
| 7891e31893 | |||
| bff74446fe | |||
| b940ad39f9 | |||
| f45f31b49c | |||
| 978e177715 | |||
| 038c92e49f | |||
| 3861d07cf4 | |||
| 41d45d1210 | |||
| cfe780c6f0 | |||
| 3ee67c8763 | |||
| edc56bc457 | |||
| 73c5fe14e7 | |||
| d9ccbfef5c | |||
| 01cac0e8e6 | |||
| 66f4a215f7 | |||
| 0bd9e88acc | |||
| f0b9bd2688 | |||
| e1629ce516 | |||
| 1006a04337 | |||
| 547af71de8 | |||
| 1b0a80052d | |||
| d38f03b8f4 | |||
| dbea15a34f | |||
| 2972297903 | |||
| 03f82d4841 | |||
| 33c2265e56 |
@@ -1,5 +1,6 @@
|
|||||||
# IDE
|
# IDE
|
||||||
.idea
|
.idea
|
||||||
|
.vscode
|
||||||
|
|
||||||
# Dependencies
|
# Dependencies
|
||||||
/node_modules
|
/node_modules
|
||||||
@@ -54,3 +55,5 @@ next-env.d.ts
|
|||||||
/pgdata
|
/pgdata
|
||||||
/test-media
|
/test-media
|
||||||
/test-data
|
/test-data
|
||||||
|
/bookdrop
|
||||||
|
dockerfile.patch
|
||||||
|
|||||||
@@ -17,6 +17,11 @@ services:
|
|||||||
- ./downloads:/downloads
|
- ./downloads:/downloads
|
||||||
- ./media:/media
|
- ./media:/media
|
||||||
|
|
||||||
|
# Book Drop: optional folder for Manual Import (Admin → audiobook → Manual Import)
|
||||||
|
# Map any host folder here and it will appear as a browsable root in the file picker.
|
||||||
|
# Example: - /path/to/your/audiobooks:/bookdrop
|
||||||
|
# - ./bookdrop:/bookdrop
|
||||||
|
|
||||||
# PostgreSQL data persistence
|
# PostgreSQL data persistence
|
||||||
- ./pgdata:/var/lib/postgresql/data
|
- ./pgdata:/var/lib/postgresql/data
|
||||||
|
|
||||||
|
|||||||
@@ -0,0 +1,335 @@
|
|||||||
|
# Documentation System Agent — Master Prompt
|
||||||
|
|
||||||
|
You are a documentation architect. Your job is to analyze a codebase from scratch and produce a **cascading, token-efficient documentation system** with a navigational index. When you are done, future AI agents dropped into this repo will be able to find any information they need by reading a single table of contents file, then following a link to exactly the right document — never wasting tokens reading irrelevant material.
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## 1. What You Are Building
|
||||||
|
|
||||||
|
You are building three things:
|
||||||
|
|
||||||
|
### A. A `documentation/` directory
|
||||||
|
A tree of concise, AI-optimized markdown files that describe every meaningful part of the codebase. The structure mirrors the codebase's own architecture (backend services, frontend components, integrations, configuration, etc.) rather than imposing an arbitrary layout.
|
||||||
|
|
||||||
|
### B. A `documentation/TABLEOFCONTENTS.md` file
|
||||||
|
The **single entry point** for all documentation. This file maps natural-language questions and topic keywords to specific documentation files. Any agent that needs to understand something reads this file first, finds the 1-3 relevant docs, and reads only those. This is the most important file you will produce.
|
||||||
|
|
||||||
|
### C. A `CLAUDE.md` file at the project root
|
||||||
|
Project instructions that teach future agents how to use the documentation system. This file is automatically loaded into every Claude Code conversation, so it must be concise, directive, and self-contained.
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## 2. The Token-Efficient Documentation Format
|
||||||
|
|
||||||
|
Every documentation file you create MUST follow this format. No exceptions.
|
||||||
|
|
||||||
|
### 2.1 Structure Template
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
# [Title]
|
||||||
|
|
||||||
|
**Status:** [Implemented | Partial | Planned] — [One-line summary of what this is]
|
||||||
|
|
||||||
|
## Overview
|
||||||
|
[1-3 sentences. What is this? What does it do? Why does it exist?]
|
||||||
|
|
||||||
|
## Key Details
|
||||||
|
- Bullet points, not prose
|
||||||
|
- Data models: field names, types, constraints
|
||||||
|
- API endpoints: method, path, request/response shape
|
||||||
|
- Config keys and their values/defaults
|
||||||
|
- Enums, status values, important constants
|
||||||
|
- File paths and code locations
|
||||||
|
- Behavioral rules and edge cases
|
||||||
|
|
||||||
|
## API / Interfaces
|
||||||
|
[If applicable — tables or compact code blocks for endpoints, function signatures, event names, etc.]
|
||||||
|
|
||||||
|
## Dependencies
|
||||||
|
[What this depends on, and what depends on it — keep to a bullet list]
|
||||||
|
|
||||||
|
## Known Issues / Gotchas
|
||||||
|
[Only if there are real, non-obvious pitfalls. Omit section entirely if none.]
|
||||||
|
|
||||||
|
## Related
|
||||||
|
- [Link to related doc 1]
|
||||||
|
- [Link to related doc 2]
|
||||||
|
```
|
||||||
|
|
||||||
|
### 2.2 Format Rules
|
||||||
|
|
||||||
|
**REQUIRED — always include:**
|
||||||
|
- Status line with one-line summary
|
||||||
|
- API endpoints, data models, config keys (complete and accurate)
|
||||||
|
- File paths to source code (so agents can navigate directly)
|
||||||
|
- Enums, constants, and status values (exact strings/numbers)
|
||||||
|
- Dependency relationships between components
|
||||||
|
- Gotchas that have caused or could cause bugs
|
||||||
|
|
||||||
|
**FORBIDDEN — never include:**
|
||||||
|
- Verbose prose or narrative explanations
|
||||||
|
- "Why we chose X" sections (brief rationale in a bullet is fine)
|
||||||
|
- ASCII art diagrams larger than 5 lines
|
||||||
|
- More than 2 code examples per document
|
||||||
|
- "Future enhancements" or roadmap speculation
|
||||||
|
- "Testing strategy" sections (unless tests are the subject of the doc)
|
||||||
|
- "Performance considerations" (unless performance is the subject)
|
||||||
|
- Empty sections or placeholder text
|
||||||
|
- Decorative formatting, horizontal rules between every section, emoji
|
||||||
|
|
||||||
|
**TARGET:** Each doc file should be 30-80 lines. If it exceeds 120 lines, split it into sub-documents and link from a parent. The goal is ~70% fewer tokens than traditional documentation while preserving 100% of the technical details an agent needs.
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## 3. The TABLEOFCONTENTS.md Format
|
||||||
|
|
||||||
|
This is the **router**. It maps questions to files. Format:
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
# Table of Contents — [Project Name]
|
||||||
|
|
||||||
|
> **Read this file first.** Find your topic below, then read ONLY the linked files.
|
||||||
|
|
||||||
|
## Quick Reference
|
||||||
|
| Topic | File |
|
||||||
|
|-------|------|
|
||||||
|
| [Short topic] | [path/to/file.md] |
|
||||||
|
| ... | ... |
|
||||||
|
|
||||||
|
## By Category
|
||||||
|
|
||||||
|
### [Category Name] (e.g., "Authentication", "Database", "API Endpoints")
|
||||||
|
| Question / Topic | File(s) |
|
||||||
|
|-------------------|---------|
|
||||||
|
| How does [X] work? | [path.md] |
|
||||||
|
| What are the [Y] endpoints? | [path.md] |
|
||||||
|
| How is [Z] configured? | [path1.md], [path2.md] |
|
||||||
|
|
||||||
|
### [Next Category]
|
||||||
|
...
|
||||||
|
|
||||||
|
## Architecture Overview
|
||||||
|
[3-10 bullet points describing the high-level architecture — frameworks, major services, data flow. Just enough for an agent to orient itself before diving into specific docs.]
|
||||||
|
```
|
||||||
|
|
||||||
|
**Rules for TABLEOFCONTENTS.md:**
|
||||||
|
- Every documentation file MUST appear in at least one table row
|
||||||
|
- Questions should be phrased the way a developer or AI agent would actually ask them
|
||||||
|
- A single question can map to multiple files (e.g., "How do downloads work?" → `downloads.md`, `jobs.md`)
|
||||||
|
- A single file can appear under multiple questions
|
||||||
|
- Categories should match the codebase's actual domain boundaries, not generic labels
|
||||||
|
- The Architecture Overview section gives agents a 30-second orientation before they search for specifics
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## 4. Execution Plan
|
||||||
|
|
||||||
|
Follow these phases in order. **Delegate heavily using the Task tool** — you should be orchestrating, not doing all the reading yourself.
|
||||||
|
|
||||||
|
### Phase 1: Deep Discovery (Delegate to Explore Agents)
|
||||||
|
|
||||||
|
Launch **3-5 parallel Explore agents** using the Task tool to map the entire codebase. Each agent should focus on a different area. Suggested splits:
|
||||||
|
|
||||||
|
**Agent 1 — Project Structure & Config:**
|
||||||
|
- Map the top-level directory tree (2-3 levels deep)
|
||||||
|
- Identify the tech stack (languages, frameworks, package managers)
|
||||||
|
- Read config files (package.json, tsconfig, docker-compose, .env.example, etc.)
|
||||||
|
- Identify build/deploy pipeline
|
||||||
|
- Note the entry points of the application
|
||||||
|
|
||||||
|
**Agent 2 — Backend / Server-Side:**
|
||||||
|
- Identify all backend services, controllers, routes, middleware
|
||||||
|
- Map API endpoints (paths, methods, handlers)
|
||||||
|
- Identify the database layer (ORM, schema files, migrations)
|
||||||
|
- Note background jobs, queues, cron tasks, workers
|
||||||
|
- Identify authentication/authorization mechanisms
|
||||||
|
|
||||||
|
**Agent 3 — Frontend / Client-Side:**
|
||||||
|
- Identify UI framework and component structure
|
||||||
|
- Map page routes and navigation
|
||||||
|
- Identify state management approach
|
||||||
|
- Note API client/service layer
|
||||||
|
- Identify shared components, layouts, hooks
|
||||||
|
|
||||||
|
**Agent 4 — Integrations & External Services:**
|
||||||
|
- Identify all third-party API integrations
|
||||||
|
- Map external service connections (databases, caches, message queues, cloud services)
|
||||||
|
- Note webhook handlers, OAuth flows, API keys
|
||||||
|
- Identify notification systems (email, push, SMS)
|
||||||
|
|
||||||
|
**Agent 5 — Data Layer & Business Logic:**
|
||||||
|
- Map database schema (tables/collections, relationships, key fields)
|
||||||
|
- Identify core business logic and domain models
|
||||||
|
- Map data validation rules
|
||||||
|
- Note important algorithms or complex logic
|
||||||
|
|
||||||
|
Adjust these splits based on what the repo actually contains. A frontend-only repo doesn't need a backend agent. A CLI tool doesn't need a frontend agent. Use your judgment.
|
||||||
|
|
||||||
|
**Each agent should return:**
|
||||||
|
- A structured summary of what it found
|
||||||
|
- File paths to the most important source files
|
||||||
|
- A suggested list of documentation topics for its area
|
||||||
|
|
||||||
|
### Phase 2: Architecture Synthesis
|
||||||
|
|
||||||
|
After all discovery agents return, synthesize their findings:
|
||||||
|
|
||||||
|
1. **Draw the dependency map** — What are the major components? How do they connect?
|
||||||
|
2. **Identify documentation topics** — Each distinct service, feature, integration, or subsystem gets its own doc file
|
||||||
|
3. **Design the directory structure** — Mirror the codebase's architecture. Example:
|
||||||
|
```
|
||||||
|
documentation/
|
||||||
|
├── TABLEOFCONTENTS.md
|
||||||
|
├── README.md # Project overview (brief)
|
||||||
|
├── architecture.md # System architecture, tech stack, data flow
|
||||||
|
├── backend/
|
||||||
|
│ ├── api-endpoints.md # Or split by domain: users.md, orders.md, etc.
|
||||||
|
│ ├── database.md # Schema, ORM, migrations
|
||||||
|
│ ├── auth.md # Authentication & authorization
|
||||||
|
│ └── jobs.md # Background processing
|
||||||
|
├── frontend/
|
||||||
|
│ ├── components.md # Component tree, shared components
|
||||||
|
│ ├── routing.md # Pages, navigation, guards
|
||||||
|
│ └── state.md # State management
|
||||||
|
├── integrations/
|
||||||
|
│ ├── [service-name].md # One per external integration
|
||||||
|
│ └── ...
|
||||||
|
└── deployment/
|
||||||
|
└── docker.md # Or whatever the deploy mechanism is
|
||||||
|
```
|
||||||
|
4. **Prioritize** — Rank topics by impact. High-impact = core architecture, APIs, database schema, auth, and anything with complex logic or non-obvious behavior. Low-impact = static config files, simple utility functions, standard boilerplate.
|
||||||
|
|
||||||
|
### Phase 3: Documentation Generation (Delegate to Writer Agents)
|
||||||
|
|
||||||
|
Launch **parallel writer agents** using the Task tool. Each agent writes 2-5 related documentation files.
|
||||||
|
|
||||||
|
**Instructions for each writer agent must include:**
|
||||||
|
- The exact file paths to create
|
||||||
|
- The list of source files to read for that topic
|
||||||
|
- The token-efficient format template (copy Section 2.1 into each agent's prompt)
|
||||||
|
- A reminder: "Write concise bullets, not prose. Include all technical details. Target 30-80 lines per file."
|
||||||
|
|
||||||
|
**Suggested batching:**
|
||||||
|
- Agent A: `architecture.md` + `README.md` (needs broadest context)
|
||||||
|
- Agent B: Backend services docs (group related services)
|
||||||
|
- Agent C: Frontend docs
|
||||||
|
- Agent D: Integration docs
|
||||||
|
- Agent E: Database + deployment docs
|
||||||
|
|
||||||
|
Scale the number of agents to the size of the repo. A small repo might need 2-3 writers. A large monorepo might need 8-10.
|
||||||
|
|
||||||
|
**Each writer agent should return:** Confirmation of files written, with a brief summary of what each file covers and a list of cross-references to note for the TOC.
|
||||||
|
|
||||||
|
### Phase 4: Build the TABLEOFCONTENTS.md
|
||||||
|
|
||||||
|
After all writers finish, build the table of contents yourself. This requires you to:
|
||||||
|
|
||||||
|
1. Read or review every documentation file that was created
|
||||||
|
2. For each file, generate 2-5 natural-language questions it answers
|
||||||
|
3. Organize questions into categories that match the codebase's domain
|
||||||
|
4. Write the Architecture Overview section (3-10 bullets, high-level only)
|
||||||
|
5. Cross-check: every doc file appears in at least one row; no dead links
|
||||||
|
|
||||||
|
### Phase 5: Generate the CLAUDE.md
|
||||||
|
|
||||||
|
Write the project-root `CLAUDE.md` using the template in Section 5 below. Customize it for this specific repo — fill in the actual project name, the actual documentation structure, and real examples from the actual TOC.
|
||||||
|
|
||||||
|
### Phase 6: Validate
|
||||||
|
|
||||||
|
Do a final pass:
|
||||||
|
1. Verify every file referenced in TABLEOFCONTENTS.md actually exists
|
||||||
|
2. Verify every file in the `documentation/` directory appears in TABLEOFCONTENTS.md
|
||||||
|
3. Spot-check 2-3 doc files for format compliance (status line, bullets not prose, within line limits)
|
||||||
|
4. Verify CLAUDE.md references the correct paths
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## 5. CLAUDE.md Template
|
||||||
|
|
||||||
|
Generate a `CLAUDE.md` at the project root using this template. **Customize every bracketed item** for the specific repo. Remove sections that don't apply. Keep it under 200 lines — this file is loaded into every conversation and consumes tokens.
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
# CLAUDE.md — [Project Name]
|
||||||
|
|
||||||
|
## Documentation System
|
||||||
|
|
||||||
|
This project uses a cascading, token-efficient documentation system optimized for AI agent consumption.
|
||||||
|
|
||||||
|
### How to Find Information
|
||||||
|
|
||||||
|
1. **Read `documentation/TABLEOFCONTENTS.md` FIRST** — this is the navigation index
|
||||||
|
2. Find your topic in the question-to-file mapping tables
|
||||||
|
3. Read ONLY the 1-3 files relevant to your task
|
||||||
|
4. **Never read all documentation files** — this wastes token budget
|
||||||
|
|
||||||
|
### Documentation Structure
|
||||||
|
[Insert the actual directory tree of documentation/ here]
|
||||||
|
|
||||||
|
### Example Lookups
|
||||||
|
- "[Example question 1]" → `[actual-path-1.md]`
|
||||||
|
- "[Example question 2]" → `[actual-path-2.md]`, `[actual-path-3.md]`
|
||||||
|
- "[Example question 3]" → `[actual-path-4.md]`
|
||||||
|
|
||||||
|
## Token Budget Rules
|
||||||
|
|
||||||
|
- **20-30% of tokens:** Reading documentation (via TABLEOFCONTENTS.md targeting)
|
||||||
|
- **70-80% of tokens:** Implementation and problem-solving
|
||||||
|
|
||||||
|
**Do:**
|
||||||
|
- Use TABLEOFCONTENTS.md to target specific files
|
||||||
|
- Read only "Key Details" and "API/Interfaces" sections
|
||||||
|
- Skip code examples unless implementing similar functionality
|
||||||
|
|
||||||
|
**Don't:**
|
||||||
|
- Read all documentation files sequentially
|
||||||
|
- Read verbose examples when not needed
|
||||||
|
- Re-read the same docs multiple times in one session
|
||||||
|
|
||||||
|
## Documentation Maintenance
|
||||||
|
|
||||||
|
When you modify code that changes behavior documented in `documentation/`:
|
||||||
|
1. Read TABLEOFCONTENTS.md to find the relevant doc(s)
|
||||||
|
2. Update those docs to reflect your changes
|
||||||
|
3. Use the token-efficient format: bullets, tables, compact code blocks — no prose
|
||||||
|
4. If you create a new doc, add it to TABLEOFCONTENTS.md
|
||||||
|
|
||||||
|
### Token-Efficient Format Reference
|
||||||
|
- **Status line:** `**Status:** [Implemented | Partial | Planned] — [one-line summary]`
|
||||||
|
- **Bullets, not paragraphs** — every detail as a dash-prefixed list item
|
||||||
|
- **Tables for APIs** — method, path, request, response
|
||||||
|
- **Code blocks only for schemas/configs** — max 2 per document
|
||||||
|
- **30-80 lines per file** — split if over 120
|
||||||
|
- **No:** prose explanations, future plans, testing strategy, empty sections
|
||||||
|
```
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## 6. Quality Standards
|
||||||
|
|
||||||
|
Your output will be evaluated on:
|
||||||
|
|
||||||
|
1. **TABLEOFCONTENTS.md completeness** — Can an agent find any topic by searching this one file?
|
||||||
|
2. **Question quality** — Are the TOC questions phrased the way someone would actually ask them?
|
||||||
|
3. **Format compliance** — Do all docs follow the token-efficient format? No prose, no fluff?
|
||||||
|
4. **Accuracy** — Do the docs match what's actually in the code? Are file paths correct?
|
||||||
|
5. **Coverage** — Are all high-impact areas documented? Are low-impact areas at least listed?
|
||||||
|
6. **CLAUDE.md clarity** — Could a brand-new agent read CLAUDE.md and immediately know how to navigate the docs?
|
||||||
|
7. **Cross-referencing** — Do Related sections link to the right companion docs?
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## 7. Important Reminders
|
||||||
|
|
||||||
|
- **You are writing for AI agents, not humans.** Optimize for parseability and token efficiency, not readability or visual appeal.
|
||||||
|
- **Accuracy over completeness.** It's better to document 80% of the codebase accurately than 100% with errors. If a discovery agent can't determine something with confidence, note it as `**Status:** Partial` and move on.
|
||||||
|
- **Mirror the codebase's language.** Use the same names for things that the code uses. If the code calls it a "processor," don't call it a "handler" in the docs.
|
||||||
|
- **File paths are critical.** Every doc should reference the actual source files it describes. Agents will use these paths to navigate directly to code.
|
||||||
|
- **The TOC is the product.** The individual doc files are supporting material. If the TOC is excellent, the whole system works. If the TOC is poor, nothing else matters.
|
||||||
|
- **Delegate aggressively.** You have access to the Task tool with sub-agents. Use it. The discovery phase should be 3-5 parallel agents. The writing phase should be 2-10 parallel agents depending on repo size. Your job is to orchestrate, synthesize, and build the TOC — not to read every file yourself.
|
||||||
|
- **Do not add headers or comments to source code files.** Your output is documentation files only. Do not modify any existing source code.
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## Now Begin
|
||||||
|
|
||||||
|
Start with Phase 1. Launch your discovery agents in parallel. Once they report back, proceed through the remaining phases. When complete, report what you've created and provide the full TABLEOFCONTENTS.md for review.
|
||||||
@@ -32,6 +32,14 @@
|
|||||||
- **File hash matching for accurate ASIN** → [fixes/file-hash-matching.md](fixes/file-hash-matching.md)
|
- **File hash matching for accurate ASIN** → [fixes/file-hash-matching.md](fixes/file-hash-matching.md)
|
||||||
- **OIDC authentication** → [backend/services/auth.md](backend/services/auth.md)
|
- **OIDC authentication** → [backend/services/auth.md](backend/services/auth.md)
|
||||||
|
|
||||||
|
## Reading Shelves (Goodreads, Hardcover)
|
||||||
|
- **Goodreads shelf sync (RSS feeds)** → [backend/services/goodreads-sync.md](backend/services/goodreads-sync.md)
|
||||||
|
- **Hardcover shelf sync (GraphQL API)** → [backend/services/hardcover-sync.md](backend/services/hardcover-sync.md)
|
||||||
|
- **Shared sync core (Audible lookup, request creation)** → [backend/services/goodreads-sync.md](backend/services/goodreads-sync.md#shared-sync-core)
|
||||||
|
- **Combined shelves API, GenericShelf** → [backend/services/goodreads-sync.md](backend/services/goodreads-sync.md)
|
||||||
|
- **Hook factory (createShelfHooks)** → [backend/services/goodreads-sync.md](backend/services/goodreads-sync.md#hook-factory)
|
||||||
|
- **Adding a new shelf provider** → [backend/services/goodreads-sync.md](backend/services/goodreads-sync.md#adding-a-new-provider)
|
||||||
|
|
||||||
## Audible Integration
|
## Audible Integration
|
||||||
- **Web scraping (popular, new releases)** → [integrations/audible.md](integrations/audible.md)
|
- **Web scraping (popular, new releases)** → [integrations/audible.md](integrations/audible.md)
|
||||||
- **Database caching, real-time matching** → [integrations/audible.md](integrations/audible.md)
|
- **Database caching, real-time matching** → [integrations/audible.md](integrations/audible.md)
|
||||||
@@ -77,6 +85,7 @@
|
|||||||
- **Component catalog (cards, badges, forms)** → [frontend/components.md](frontend/components.md)
|
- **Component catalog (cards, badges, forms)** → [frontend/components.md](frontend/components.md)
|
||||||
- **RequestCard, StatusBadge, ProgressBar** → [frontend/components.md](frontend/components.md)
|
- **RequestCard, StatusBadge, ProgressBar** → [frontend/components.md](frontend/components.md)
|
||||||
- **Pages: home, search, requests, profile** → [frontend/components.md](frontend/components.md)
|
- **Pages: home, search, requests, profile** → [frontend/components.md](frontend/components.md)
|
||||||
|
- **Home page sections (per-user, configurable)** → [features/home-sections.md](features/home-sections.md)
|
||||||
|
|
||||||
## BookDate (AI Recommendations)
|
## BookDate (AI Recommendations)
|
||||||
- **AI-powered recommendations, swipe interface** → [features/bookdate.md](features/bookdate.md)
|
- **AI-powered recommendations, swipe interface** → [features/bookdate.md](features/bookdate.md)
|
||||||
@@ -150,3 +159,10 @@
|
|||||||
**"Why do BookDate library books show placeholders?"** → [features/library-thumbnail-cache.md](features/library-thumbnail-cache.md)
|
**"Why do BookDate library books show placeholders?"** → [features/library-thumbnail-cache.md](features/library-thumbnail-cache.md)
|
||||||
**"How does file hash matching work?"** → [fixes/file-hash-matching.md](fixes/file-hash-matching.md)
|
**"How does file hash matching work?"** → [fixes/file-hash-matching.md](fixes/file-hash-matching.md)
|
||||||
**"Why is ABS matching the wrong book?"** → [fixes/file-hash-matching.md](fixes/file-hash-matching.md) (file hash prevents false positives)
|
**"Why is ABS matching the wrong book?"** → [fixes/file-hash-matching.md](fixes/file-hash-matching.md) (file hash prevents false positives)
|
||||||
|
**"How do I customize my home page?"** → [features/home-sections.md](features/home-sections.md)
|
||||||
|
**"How do Audible categories work?"** → [features/home-sections.md](features/home-sections.md)
|
||||||
|
**"How do I add category sections to the home page?"** → [features/home-sections.md](features/home-sections.md)
|
||||||
|
**"How do Goodreads shelves work?"** → [backend/services/goodreads-sync.md](backend/services/goodreads-sync.md)
|
||||||
|
**"How do Hardcover shelves work?"** → [backend/services/hardcover-sync.md](backend/services/hardcover-sync.md)
|
||||||
|
**"How do I add a new shelf provider?"** → [backend/services/goodreads-sync.md](backend/services/goodreads-sync.md#adding-a-new-provider)
|
||||||
|
**"How does the shelf sync core work?"** → [backend/services/goodreads-sync.md](backend/services/goodreads-sync.md#shared-sync-core)
|
||||||
|
|||||||
@@ -0,0 +1,75 @@
|
|||||||
|
# Goodreads & Shelf Sync
|
||||||
|
|
||||||
|
**Status:** ✅ Implemented | RSS feed parsing, shared sync core, extensible provider architecture
|
||||||
|
|
||||||
|
## Overview
|
||||||
|
Syncs user-subscribed Goodreads shelves via RSS feeds, resolves books to Audible ASINs, and creates requests. Also documents the shared shelf sync core used by all providers.
|
||||||
|
|
||||||
|
## Architecture
|
||||||
|
|
||||||
|
### Files
|
||||||
|
- `src/lib/services/goodreads-sync.service.ts` — RSS fetch/parse, delegates to shared core
|
||||||
|
- `src/lib/services/shelf-sync-core.service.ts` — Shared sync logic (Audible lookup, cover enrichment, request creation)
|
||||||
|
- `src/lib/utils/shelf-helpers.ts` — Shared `processBooks()` utility for cover URL parsing
|
||||||
|
- `src/lib/hooks/createShelfHooks.ts` — Generic hook factory for shelf CRUD operations
|
||||||
|
- `src/app/api/user/goodreads-shelves/route.ts` — GET (list) + POST (add) routes
|
||||||
|
- `src/app/api/user/goodreads-shelves/[id]/route.ts` — DELETE + PATCH routes
|
||||||
|
- `src/app/api/user/shelves/route.ts` — Combined GET for all providers (GenericShelf shape)
|
||||||
|
- `src/lib/hooks/useGoodreadsShelves.ts` — Frontend hooks (via `createShelfHooks` factory)
|
||||||
|
|
||||||
|
### Database Models
|
||||||
|
- **GoodreadsShelf** — Per-user shelf subscription (`userId`, `rssUrl`, `name`, `lastSyncAt`, `bookCount`, `coverUrls`)
|
||||||
|
- **BookMapping** — Shared table for all providers. Keyed by `provider` + `externalBookId`. Caches Audible ASIN lookups.
|
||||||
|
|
||||||
|
## Goodreads RSS Feed
|
||||||
|
- **Format:** `https://www.goodreads.com/review/list_rss/{userId}?shelf={shelfName}`
|
||||||
|
- **Auth:** None required (public RSS)
|
||||||
|
- **Parsing:** `fast-xml-parser` extracts `item` entries with `book_id`, `title`, `author_name`, `book_image_url`
|
||||||
|
|
||||||
|
## Shared Sync Core
|
||||||
|
|
||||||
|
`shelf-sync-core.service.ts` contains all provider-agnostic sync logic:
|
||||||
|
|
||||||
|
### Interface: `ShelfBook`
|
||||||
|
```typescript
|
||||||
|
{ bookId: string; title: string; author: string; coverUrl?: string }
|
||||||
|
```
|
||||||
|
|
||||||
|
### Function: `processShelfBooks()`
|
||||||
|
Accepts provider-agnostic book list + context, performs:
|
||||||
|
1. **BookMapping lookup** — Check if book already resolved (`provider` + `externalBookId`)
|
||||||
|
2. **Audible search** — Full query (`title author`), fallback with cleaned title (strips parenthetical series info)
|
||||||
|
3. **noMatch retry** — Re-searches after `NO_MATCH_RETRY_DAYS` (7 days)
|
||||||
|
4. **Request creation** — Calls `createRequestForUser()` for matched ASINs
|
||||||
|
5. **Cover enrichment** — Queries `audibleCache` for cached covers, builds `/api/cache/thumbnails/` URLs
|
||||||
|
6. **Shelf metadata update** — Writes `lastSyncAt`, `bookCount`, top 8 books as JSON to `coverUrls`
|
||||||
|
|
||||||
|
### Constants
|
||||||
|
- `DEFAULT_MAX_LOOKUPS_PER_SHELF` = 10 (per scheduled cycle; 0 = unlimited for manual triggers)
|
||||||
|
- `NO_MATCH_RETRY_DAYS` = 7
|
||||||
|
|
||||||
|
### Hook Factory: `createShelfHooks(endpoint)`
|
||||||
|
Returns `{ useList, useAdd, useDelete, useUpdate }` — all with SWR caching, optimistic updates, and automatic revalidation of the combined `/api/user/shelves` endpoint.
|
||||||
|
|
||||||
|
## API Endpoints
|
||||||
|
|
||||||
|
| Method | Path | Purpose |
|
||||||
|
|---|---|---|
|
||||||
|
| GET | `/api/user/goodreads-shelves` | List user's Goodreads shelves |
|
||||||
|
| POST | `/api/user/goodreads-shelves` | Add shelf (validates RSS feed, triggers sync) |
|
||||||
|
| DELETE | `/api/user/goodreads-shelves/[id]` | Remove shelf (ownership check) |
|
||||||
|
| PATCH | `/api/user/goodreads-shelves/[id]` | Update RSS URL (triggers re-sync) |
|
||||||
|
| GET | `/api/user/shelves` | Combined endpoint — merges all providers into `GenericShelf` |
|
||||||
|
|
||||||
|
## Adding a New Provider
|
||||||
|
1. Create Prisma shelf model + migration (BookMapping table is already shared)
|
||||||
|
2. Create API client service for the external data source
|
||||||
|
3. Create thin sync service (~50-80 lines) that fetches books and calls `processShelfBooks()`
|
||||||
|
4. Create API routes (or use a generic route handler)
|
||||||
|
5. Create hook file (~40 lines) using `createShelfHooks(endpoint)`
|
||||||
|
6. Add tab in `AddShelfModal` with provider-specific form fields
|
||||||
|
|
||||||
|
## Related
|
||||||
|
- [Hardcover sync](hardcover-sync.md)
|
||||||
|
- [Background jobs](jobs.md)
|
||||||
|
- [Scheduler](scheduler.md)
|
||||||
@@ -0,0 +1,66 @@
|
|||||||
|
# Hardcover Shelf Sync
|
||||||
|
|
||||||
|
**Status:** ✅ Implemented | GraphQL API integration, Audible ASIN resolution, automated request creation
|
||||||
|
|
||||||
|
## Overview
|
||||||
|
Syncs user-subscribed Hardcover lists via their GraphQL API, resolves books to Audible ASINs, and creates audiobook requests automatically.
|
||||||
|
|
||||||
|
## Architecture
|
||||||
|
|
||||||
|
### Files
|
||||||
|
- `src/lib/services/hardcover-api.service.ts` — GraphQL queries, `fetchHardcoverList()`
|
||||||
|
- `src/lib/services/hardcover-sync.service.ts` — Provider-specific orchestration, delegates to shared core
|
||||||
|
- `src/lib/services/shelf-sync-core.service.ts` — Shared sync logic (Audible lookup, cover enrichment, request creation)
|
||||||
|
- `src/app/api/user/hardcover-shelves/route.ts` — GET (list) + POST (add) routes
|
||||||
|
- `src/app/api/user/hardcover-shelves/[id]/route.ts` — DELETE + PATCH routes
|
||||||
|
- `src/lib/hooks/useHardcoverShelves.ts` — Frontend hooks (via `createShelfHooks` factory)
|
||||||
|
|
||||||
|
### Database Models
|
||||||
|
- **HardcoverShelf** — Per-user list subscription (`userId`, `listId`, encrypted `apiToken`, `name`, `lastSyncAt`, `bookCount`, `coverUrls`)
|
||||||
|
- **BookMapping** — Shared across all providers. Keyed by `provider` + `externalBookId`. Caches Audible ASIN resolution (`audibleAsin`, `noMatch`, `lastSearchAt`)
|
||||||
|
|
||||||
|
## Hardcover API
|
||||||
|
|
||||||
|
- **Endpoint:** `https://api.hardcover.app/v1/graphql` (Hasura-based)
|
||||||
|
- **Auth:** Bearer token in Authorization header
|
||||||
|
- **Username type:** `citext` (case-insensitive text) — use `$username: citext!` in GraphQL variables
|
||||||
|
|
||||||
|
### Query Strategies (custom lists)
|
||||||
|
| Input | Strategy | Query root |
|
||||||
|
|---|---|---|
|
||||||
|
| URL with `@username` | Scoped to that user | `users(where: {username: {_eq: $username}}) { lists(...) }` |
|
||||||
|
| Bare slug (no username) | Authenticated user's own list | `me { lists(where: {slug: {_eq: $slug}}) }` |
|
||||||
|
| Numeric ID | Global lookup (IDs are unique) | `lists(where: {id: {_eq: $listId}})` |
|
||||||
|
|
||||||
|
### Status Lists
|
||||||
|
- Prefix: `status-{id}` (e.g., `status-1`)
|
||||||
|
- Query: `me { user_books(where: {status_id: {_eq: $statusId}}) }`
|
||||||
|
- Status IDs: 1=Want to Read, 2=Currently Reading, 3=Read, 4=Did Not Finish
|
||||||
|
|
||||||
|
## Sync Flow
|
||||||
|
1. Fetch shelves from DB (all or specific `shelfId`)
|
||||||
|
2. Decrypt API token (encryption service)
|
||||||
|
3. Fetch books from Hardcover GraphQL API
|
||||||
|
4. Delegate to `processShelfBooks()` in shelf-sync-core (Audible lookup, request creation, cover enrichment)
|
||||||
|
5. Update shelf metadata (`lastSyncAt`, `bookCount`, `coverUrls`)
|
||||||
|
|
||||||
|
## API Endpoints
|
||||||
|
|
||||||
|
| Method | Path | Purpose |
|
||||||
|
|---|---|---|
|
||||||
|
| GET | `/api/user/hardcover-shelves` | List user's shelves with book counts/covers |
|
||||||
|
| POST | `/api/user/hardcover-shelves` | Add new shelf (validates via API fetch, encrypts token, triggers sync) |
|
||||||
|
| DELETE | `/api/user/hardcover-shelves/[id]` | Remove shelf (ownership check) |
|
||||||
|
| PATCH | `/api/user/hardcover-shelves/[id]` | Update listId/apiToken (triggers re-sync on change) |
|
||||||
|
|
||||||
|
## Key Details
|
||||||
|
- **Token cleanup:** Strips `Bearer ` prefix if user pastes it
|
||||||
|
- **Duplicate check:** Unique constraint on `(userId, listId)`
|
||||||
|
- **Immediate sync:** POST and PATCH trigger `addSyncShelvesJob()` with unlimited lookups
|
||||||
|
- **Scheduled sync:** Runs via `sync_reading_shelves` job (default: max 10 lookups/shelf/cycle)
|
||||||
|
- **Cover data:** Stores top 8 books as JSON in `coverUrls` field for shelf card display
|
||||||
|
|
||||||
|
## Related
|
||||||
|
- [Shelf sync core (shared logic)](goodreads-sync.md#shared-sync-core)
|
||||||
|
- [Background jobs](jobs.md)
|
||||||
|
- [Scheduler](scheduler.md)
|
||||||
@@ -129,10 +129,10 @@ interface ScheduledJob {
|
|||||||
## Audible Refresh Processor
|
## Audible Refresh Processor
|
||||||
|
|
||||||
**Implementation:**
|
**Implementation:**
|
||||||
1. Clear previous `isPopular`/`isNewRelease` flags
|
1. Fetch 200 popular + 200 new releases (multi-page scraping)
|
||||||
2. Fetch 200 popular + 200 new releases (multi-page scraping)
|
2. Download and cache cover thumbnails locally (stored in `/app/cache/thumbnails`)
|
||||||
3. Download and cache cover thumbnails locally (stored in `/app/cache/thumbnails`)
|
3. Wipe and re-populate `AudibleCacheCategory` entries with reserved IDs (`__popular__`, `__new_releases__`) and user-configured category IDs
|
||||||
4. Store/update in DB with category flags, rankings (`popularRank`, `newReleaseRank`), and cached cover paths
|
4. Upsert book metadata in `AudibleCache`, ranked entries in `AudibleCacheCategory`
|
||||||
5. Record sync timestamp (`lastAudibleSync`)
|
5. Record sync timestamp (`lastAudibleSync`)
|
||||||
6. Clean up unused thumbnails (removes covers for audiobooks no longer in cache)
|
6. Clean up unused thumbnails (removes covers for audiobooks no longer in cache)
|
||||||
7. Perform fuzzy matching (70% threshold) against Plex library
|
7. Perform fuzzy matching (70% threshold) against Plex library
|
||||||
|
|||||||
@@ -0,0 +1,64 @@
|
|||||||
|
# Home Page Sections (Per-User Configurable)
|
||||||
|
|
||||||
|
**Status:** Implemented | Per-user home page with configurable sections (popular, new releases, Audible categories)
|
||||||
|
|
||||||
|
## Overview
|
||||||
|
Users customize their home page by adding/removing/reordering sections. Each section displays audiobooks from a specific source: built-in Popular, New Releases, or scraped Audible categories.
|
||||||
|
|
||||||
|
## Data Models
|
||||||
|
|
||||||
|
**UserHomeSection** (`user_home_sections`):
|
||||||
|
- `id`, `userId` (FK User), `sectionType` ('popular'|'new_releases'|'category'), `categoryId` (nullable), `categoryName` (nullable), `sortOrder` (int)
|
||||||
|
- Unique: `(userId, sectionType, categoryId)`
|
||||||
|
- Default: Popular (0) + New Releases (1) created on first access
|
||||||
|
|
||||||
|
**AudibleCacheCategory** (`audible_cache_categories`):
|
||||||
|
- `id`, `asin`, `categoryId`, `rank`, `lastSyncedAt`
|
||||||
|
- Unique: `(asin, categoryId)`, Indexes: `categoryId`, `(categoryId, rank)`
|
||||||
|
|
||||||
|
## API Endpoints
|
||||||
|
|
||||||
|
| Method | Path | Auth | Description |
|
||||||
|
|--------|------|------|-------------|
|
||||||
|
| GET | `/api/user/home-sections` | user | Returns sections + nextRefresh |
|
||||||
|
| PUT | `/api/user/home-sections` | user | Save full config (delete-recreate), max 10 |
|
||||||
|
| GET | `/api/audible/categories` | user | Live scrape top-level categories |
|
||||||
|
| GET | `/api/audiobooks/category/[categoryId]` | public | Paginated category books from cache |
|
||||||
|
|
||||||
|
## Refresh Processor (Unified Storage)
|
||||||
|
- All section data stored in `AudibleCacheCategory` with reserved IDs: `__popular__` and `__new_releases__` for built-in sections
|
||||||
|
- Popular/new-releases use same wipe-and-populate pattern as user categories
|
||||||
|
- After built-in sections, queries DISTINCT categoryIds from `UserHomeSection`
|
||||||
|
- Per section: wipe `AudibleCacheCategory` rows, scrape, upsert `AudibleCache` metadata, insert ranked category entries
|
||||||
|
- Batch cooldown between sections (10-20s random)
|
||||||
|
- Constants exported from `audible-refresh.processor.ts`: `POPULAR_CATEGORY_ID`, `NEW_RELEASES_CATEGORY_ID`
|
||||||
|
|
||||||
|
## AudibleService Methods
|
||||||
|
- `getCategories()`: Scrapes `{baseUrl}/categories`, returns `{id, name}[]`
|
||||||
|
- `getCategoryBooks(categoryId, limit)`: Scrapes `/search?node={id}&pageSize=50&sort=popularity-rank`, up to 200 results
|
||||||
|
|
||||||
|
## Frontend
|
||||||
|
- **Hooks:** `useHomeSections()`, `useCategoryAudiobooks()`, `useAudibleCategories()` in `src/lib/hooks/useHomeSections.ts`
|
||||||
|
- **Config Modal:** `src/components/home/HomeSectionConfigModal.tsx` — drag-and-drop (desktop), up/down arrows (mobile), auto-save with debounce
|
||||||
|
- **Section Component:** `src/components/home/HomeSection.tsx` — renders individual section with color-coded header
|
||||||
|
- **Home Page:** `src/app/page.tsx` — dynamic sections from user config, gear icon for customize
|
||||||
|
- **Pagination:** `src/components/ui/UnifiedPagination.tsx` — updated to support 1-12 dynamic sections
|
||||||
|
|
||||||
|
## Key Decisions
|
||||||
|
- 10 section limit per user (total)
|
||||||
|
- Category picker scraped live (no categories table)
|
||||||
|
- Top-level categories only (v1)
|
||||||
|
- Wipe-and-re-scrape per category during refresh
|
||||||
|
- Deduplication of categories across users before scraping
|
||||||
|
- If category disappears, user sees empty section
|
||||||
|
- 10-color palette assigned by sort order
|
||||||
|
|
||||||
|
## Files
|
||||||
|
- Schema: `prisma/schema.prisma` (UserHomeSection, AudibleCacheCategory)
|
||||||
|
- Migration: `prisma/migrations/20260306000000_add_home_sections/migration.sql`
|
||||||
|
- Service: `src/lib/integrations/audible.service.ts` (getCategories, getCategoryBooks)
|
||||||
|
- Processor: `src/lib/processors/audible-refresh.processor.ts`
|
||||||
|
- API Routes: `src/app/api/user/home-sections/route.ts`, `src/app/api/audible/categories/route.ts`, `src/app/api/audiobooks/category/[categoryId]/route.ts`
|
||||||
|
- Hooks: `src/lib/hooks/useHomeSections.ts`
|
||||||
|
- Components: `src/components/home/HomeSectionConfigModal.tsx`, `src/components/home/HomeSection.tsx`
|
||||||
|
- Tests: `tests/api/home-sections.routes.test.ts`, `tests/processors/audible-refresh.processor.test.ts`
|
||||||
@@ -0,0 +1,87 @@
|
|||||||
|
# Manual Import Feature — Acceptance Criteria
|
||||||
|
|
||||||
|
**Status:** ⏳ In Progress
|
||||||
|
|
||||||
|
## Overview
|
||||||
|
Allow admins to manually import audiobook files from the server filesystem into RMAB's processing pipeline for a specific book.
|
||||||
|
|
||||||
|
## Acceptance Criteria
|
||||||
|
|
||||||
|
### AC-1: Manual Import Button (Frontend)
|
||||||
|
- [ ] "Manual Import" button visible on `AudiobookDetailsModal` for admin users only
|
||||||
|
- [ ] Button hidden when book is in active processing states: `downloading`, `processing`, `searching`
|
||||||
|
- [ ] Button uses `FolderArrowDownIcon` from Heroicons
|
||||||
|
- [ ] Clicking opens the file browser modal
|
||||||
|
|
||||||
|
### AC-2: File Browser Modal — Phase 1 (Browse)
|
||||||
|
- [ ] Modal opens at `max-w-2xl`, rounded-2xl, with header/breadcrumb/listing/footer regions
|
||||||
|
- [ ] Root view shows two entry tiles: Downloads and Media Library (paths from `download_dir` and `media_dir` config)
|
||||||
|
- [ ] Each folder row shows: folder icon, name, metadata line (audio file count, subfolder count, total size)
|
||||||
|
- [ ] Blue `♪ N` badge on folders containing audio files
|
||||||
|
- [ ] Folder icon swaps to `FolderOpenIcon` on hover (150ms transition)
|
||||||
|
- [ ] Single-click selects folder (only if it has audio files); double-click navigates into it
|
||||||
|
- [ ] Folders without audio files shown at reduced opacity, still navigable but not selectable
|
||||||
|
- [ ] Breadcrumb navigation with clickable segments, home icon for root, ellipsis collapse for deep paths
|
||||||
|
- [ ] Footer shows selected path (monospace), file stats, "Review Import →" button (only when valid selection)
|
||||||
|
- [ ] Directional slide animations: right when going deeper, left when going back
|
||||||
|
- [ ] Loading skeletons during directory fetch
|
||||||
|
- [ ] Empty state for empty directories
|
||||||
|
- [ ] Error state with "Try Again" for failed directory reads
|
||||||
|
- [ ] Dark mode support throughout
|
||||||
|
|
||||||
|
### AC-3: File Browser Modal — Phase 2 (Confirm)
|
||||||
|
- [ ] Slide transition from browse to confirm phase
|
||||||
|
- [ ] Shows book context: cover thumbnail + title + author
|
||||||
|
- [ ] Shows selected folder: path (monospace) + stats in inset block
|
||||||
|
- [ ] Numbered "What will happen" list: (1) copy to media library, (2) tag metadata, (3) download cover art, (4) scan library
|
||||||
|
- [ ] "Back" button returns to browse phase
|
||||||
|
- [ ] "Start Import" primary button triggers the import
|
||||||
|
- [ ] Button shows loading state during API call
|
||||||
|
- [ ] Success: close modal, show success toast, trigger request list refresh
|
||||||
|
- [ ] Error: show error toast, stay on confirm screen
|
||||||
|
|
||||||
|
### AC-4: Filesystem Browse API
|
||||||
|
- [ ] `GET /api/admin/filesystem/browse?path=...` — admin-only endpoint
|
||||||
|
- [ ] Returns directory listing: `{ entries: [{ name, type, audioFileCount, subfolderCount, totalSize }] }`
|
||||||
|
- [ ] If no `path` param, returns root directories (download_dir, media_dir from config)
|
||||||
|
- [ ] Path validation: must be within allowed root directories (prevent directory traversal)
|
||||||
|
- [ ] Handles permission errors gracefully
|
||||||
|
- [ ] Sorts: folders first, then alphabetical
|
||||||
|
|
||||||
|
### AC-5: Manual Import API
|
||||||
|
- [ ] `POST /api/admin/manual-import` — admin-only endpoint
|
||||||
|
- [ ] Request body: `{ audiobookId: string, folderPath: string }`
|
||||||
|
- [ ] Path validation: folderPath must be within allowed roots
|
||||||
|
- [ ] Validates folder exists and contains audio files
|
||||||
|
- [ ] If no existing request: creates request (status: `processing`) + queues `organize_files` job
|
||||||
|
- [ ] If existing request (non-active state): updates status to `processing` + queues `organize_files` job
|
||||||
|
- [ ] Returns: `{ success: true, requestId: string }`
|
||||||
|
- [ ] Proper error responses for: invalid path, no audio files, already processing, book not found
|
||||||
|
|
||||||
|
### AC-6: Integration with Existing Pipeline
|
||||||
|
- [ ] The `organize_files` job processes the manual import folder identically to download-client-delivered folders
|
||||||
|
- [ ] Files are copied (not moved) to the media library
|
||||||
|
- [ ] Metadata tagging, cover art download, file hash generation all work as normal
|
||||||
|
- [ ] Library scan triggered after organization (if configured)
|
||||||
|
- [ ] Request status progresses: processing → downloaded → available (via scheduled scan)
|
||||||
|
|
||||||
|
### AC-7: Docker Build
|
||||||
|
- [ ] `docker compose build readmeabook` succeeds with no errors
|
||||||
|
|
||||||
|
## Non-Goals
|
||||||
|
- No "move" option (copy only, matching existing pipeline)
|
||||||
|
- No file-level selection (folder only)
|
||||||
|
- No drag-and-drop upload
|
||||||
|
- No non-admin access
|
||||||
|
|
||||||
|
## Technical Notes
|
||||||
|
- Audio extensions: `.m4b`, `.m4a`, `.mp3`, `.mp4`, `.aa`, `.aax`, `.flac`, `.ogg` (from `src/lib/constants/audio-formats.ts`)
|
||||||
|
- Config keys: `download_dir` (database), `media_dir` (database)
|
||||||
|
- Existing file organizer: `src/lib/utils/file-organizer.ts`
|
||||||
|
- Organize processor: `src/lib/processors/organize-files.processor.ts`
|
||||||
|
- Job queue service: `src/lib/services/job-queue.service.ts`
|
||||||
|
- Auth middleware: `requireAuth()`, `requireAdmin()` from `src/lib/middleware/auth.ts`
|
||||||
|
- Frontend API pattern: `fetchWithAuth()` from `src/lib/utils/api.ts`
|
||||||
|
- Modal base: `src/components/ui/Modal.tsx`
|
||||||
|
- Audiobook details modal: `src/components/audiobooks/AudiobookDetailsModal.tsx`
|
||||||
|
- Toast: `useToast()` from toast context
|
||||||
@@ -128,11 +128,11 @@ Single matching algorithm used everywhere (search, popular, new-releases, jobs).
|
|||||||
Discovery APIs serve cached data from DB with real-time matching.
|
Discovery APIs serve cached data from DB with real-time matching.
|
||||||
|
|
||||||
**Flow:**
|
**Flow:**
|
||||||
1. `audible_refresh` job runs daily → fetches 200 popular + 200 new releases
|
1. `audible_refresh` job runs daily → fetches 200 popular + 200 new releases + user-configured categories
|
||||||
2. Downloads and caches cover thumbnails locally (reduces Audible load)
|
2. Downloads and caches cover thumbnails locally (reduces Audible load)
|
||||||
3. Stores in DB with flags (`isPopular`, `isNewRelease`) and rankings
|
3. Stores metadata in `audible_cache`, ranked entries in `audible_cache_categories` with reserved IDs (`__popular__`, `__new_releases__`) and user category IDs
|
||||||
4. Cleans up unused thumbnails after sync
|
4. Cleans up unused thumbnails after sync
|
||||||
5. API routes query DB → apply real-time matching → return enriched results
|
5. API routes query `AudibleCacheCategory` by categoryId → join with `AudibleCache` metadata → apply real-time matching → return enriched results
|
||||||
6. Homepage loads instantly (no Audible API hits)
|
6. Homepage loads instantly (no Audible API hits)
|
||||||
|
|
||||||
## Thumbnail Caching
|
## Thumbnail Caching
|
||||||
|
|||||||
@@ -51,7 +51,7 @@ Ebooks are first-class citizens in RMAB, with their own request type, tracking,
|
|||||||
| Key | Default | Description |
|
| Key | Default | Description |
|
||||||
|-----|---------|-------------|
|
|-----|---------|-------------|
|
||||||
| `ebook_annas_archive_enabled` | `false` | Enable Anna's Archive downloads |
|
| `ebook_annas_archive_enabled` | `false` | Enable Anna's Archive downloads |
|
||||||
| `ebook_sidecar_base_url` | `https://annas-archive.li` | Base URL for mirror |
|
| `ebook_sidecar_base_url` | `https://annas-archive.gl` | Base URL for mirror |
|
||||||
| `ebook_sidecar_flaresolverr_url` | `` (empty) | FlareSolverr proxy URL (optional) |
|
| `ebook_sidecar_flaresolverr_url` | `` (empty) | FlareSolverr proxy URL (optional) |
|
||||||
|
|
||||||
#### Section 2: Indexer Search
|
#### Section 2: Indexer Search
|
||||||
@@ -180,18 +180,18 @@ Configure URL in Admin Settings → E-book Sidecar: `http://localhost:8191`
|
|||||||
|
|
||||||
### Method 1: ASIN Search (exact match)
|
### Method 1: ASIN Search (exact match)
|
||||||
```
|
```
|
||||||
Search: https://annas-archive.li/search?ext=epub&lang=en&q="asin:B09TWSRMCB"
|
Search: https://annas-archive.gl/search?ext=epub&lang=en&q="asin:B09TWSRMCB"
|
||||||
↓
|
↓
|
||||||
MD5 Page: https://annas-archive.li/md5/[md5]
|
MD5 Page: https://annas-archive.gl/md5/[md5]
|
||||||
↓
|
↓
|
||||||
Slow Download: https://annas-archive.li/slow_download/[md5]/0/5
|
Slow Download: https://annas-archive.gl/slow_download/[md5]/0/5
|
||||||
↓
|
↓
|
||||||
File Server: http://[server]/path/to/file.epub
|
File Server: http://[server]/path/to/file.epub
|
||||||
```
|
```
|
||||||
|
|
||||||
### Method 2: Title + Author (fallback)
|
### Method 2: Title + Author (fallback)
|
||||||
```
|
```
|
||||||
Search: https://annas-archive.li/search?q=Title+Author&ext=epub&lang=en
|
Search: https://annas-archive.gl/search?q=Title+Author&ext=epub&lang=en
|
||||||
↓ (Same flow from MD5 page)
|
↓ (Same flow from MD5 page)
|
||||||
```
|
```
|
||||||
|
|
||||||
|
|||||||
@@ -81,7 +81,7 @@ src/app/admin/settings/
|
|||||||
|
|
||||||
1. **Anna's Archive Section**
|
1. **Anna's Archive Section**
|
||||||
- Enable toggle for Anna's Archive downloads
|
- Enable toggle for Anna's Archive downloads
|
||||||
- Base URL (default: `https://annas-archive.li`)
|
- Base URL (default: `https://annas-archive.gl`)
|
||||||
- FlareSolverr URL (optional, for Cloudflare bypass)
|
- FlareSolverr URL (optional, for Cloudflare bypass)
|
||||||
|
|
||||||
2. **Indexer Search Section**
|
2. **Indexer Search Section**
|
||||||
@@ -101,7 +101,7 @@ src/app/admin/settings/
|
|||||||
| `ebook_sidecar_preferred_format` | `epub` | Preferred format |
|
| `ebook_sidecar_preferred_format` | `epub` | Preferred format |
|
||||||
| `ebook_auto_grab_enabled` | `true` | Auto-create ebook requests after audiobook downloads |
|
| `ebook_auto_grab_enabled` | `true` | Auto-create ebook requests after audiobook downloads |
|
||||||
| `ebook_kindle_fix_enabled` | `false` | Apply Kindle compatibility fixes to EPUB files |
|
| `ebook_kindle_fix_enabled` | `false` | Apply Kindle compatibility fixes to EPUB files |
|
||||||
| `ebook_sidecar_base_url` | `https://annas-archive.li` | Anna's Archive mirror |
|
| `ebook_sidecar_base_url` | `https://annas-archive.gl` | Anna's Archive mirror |
|
||||||
| `ebook_sidecar_flaresolverr_url` | `` | FlareSolverr URL |
|
| `ebook_sidecar_flaresolverr_url` | `` | FlareSolverr URL |
|
||||||
|
|
||||||
**Behavior:**
|
**Behavior:**
|
||||||
|
|||||||
Generated
+816
-27
File diff suppressed because it is too large
Load Diff
+4
-2
@@ -1,6 +1,6 @@
|
|||||||
{
|
{
|
||||||
"name": "readmeabook",
|
"name": "readmeabook",
|
||||||
"version": "1.0.11",
|
"version": "1.1.0",
|
||||||
"private": true,
|
"private": true,
|
||||||
"scripts": {
|
"scripts": {
|
||||||
"dev": "next dev",
|
"dev": "next dev",
|
||||||
@@ -18,7 +18,9 @@
|
|||||||
"dependencies": {
|
"dependencies": {
|
||||||
"@heroicons/react": "^2.2.0",
|
"@heroicons/react": "^2.2.0",
|
||||||
"@prisma/client": "^6.19.0",
|
"@prisma/client": "^6.19.0",
|
||||||
|
"@types/archiver": "^7.0.0",
|
||||||
"adm-zip": "^0.5.16",
|
"adm-zip": "^0.5.16",
|
||||||
|
"archiver": "^7.0.1",
|
||||||
"axios": "^1.7.2",
|
"axios": "^1.7.2",
|
||||||
"bcrypt": "^5.1.1",
|
"bcrypt": "^5.1.1",
|
||||||
"bull": "^4.12.0",
|
"bull": "^4.12.0",
|
||||||
@@ -43,9 +45,9 @@
|
|||||||
"devDependencies": {
|
"devDependencies": {
|
||||||
"@tailwindcss/postcss": "^4",
|
"@tailwindcss/postcss": "^4",
|
||||||
"@testing-library/jest-dom": "^6.9.1",
|
"@testing-library/jest-dom": "^6.9.1",
|
||||||
"@types/adm-zip": "^0.5.6",
|
|
||||||
"@testing-library/react": "^16.3.1",
|
"@testing-library/react": "^16.3.1",
|
||||||
"@testing-library/user-event": "^14.6.1",
|
"@testing-library/user-event": "^14.6.1",
|
||||||
|
"@types/adm-zip": "^0.5.6",
|
||||||
"@types/bcrypt": "^5.0.2",
|
"@types/bcrypt": "^5.0.2",
|
||||||
"@types/bull": "^4.10.0",
|
"@types/bull": "^4.10.0",
|
||||||
"@types/jsonwebtoken": "^9.0.6",
|
"@types/jsonwebtoken": "^9.0.6",
|
||||||
|
|||||||
@@ -0,0 +1,2 @@
|
|||||||
|
-- AlterTable
|
||||||
|
ALTER TABLE "users" ADD COLUMN "download_access" BOOLEAN;
|
||||||
@@ -0,0 +1,2 @@
|
|||||||
|
-- AlterTable
|
||||||
|
ALTER TABLE "requests" ADD COLUMN "custom_search_terms" TEXT;
|
||||||
@@ -0,0 +1,42 @@
|
|||||||
|
-- CreateTable
|
||||||
|
CREATE TABLE "works" (
|
||||||
|
"id" TEXT NOT NULL,
|
||||||
|
"title" TEXT NOT NULL,
|
||||||
|
"author" TEXT NOT NULL,
|
||||||
|
"created_at" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
|
||||||
|
"updated_at" TIMESTAMP(3) NOT NULL,
|
||||||
|
|
||||||
|
CONSTRAINT "works_pkey" PRIMARY KEY ("id")
|
||||||
|
);
|
||||||
|
|
||||||
|
-- CreateTable
|
||||||
|
CREATE TABLE "work_asins" (
|
||||||
|
"id" TEXT NOT NULL,
|
||||||
|
"work_id" TEXT NOT NULL,
|
||||||
|
"asin" TEXT NOT NULL,
|
||||||
|
"narrator" TEXT,
|
||||||
|
"duration_minutes" INTEGER,
|
||||||
|
"is_canonical" BOOLEAN NOT NULL DEFAULT false,
|
||||||
|
"source" TEXT NOT NULL,
|
||||||
|
"created_at" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
|
||||||
|
|
||||||
|
CONSTRAINT "work_asins_pkey" PRIMARY KEY ("id")
|
||||||
|
);
|
||||||
|
|
||||||
|
-- CreateIndex
|
||||||
|
CREATE INDEX "works_title_idx" ON "works"("title");
|
||||||
|
|
||||||
|
-- CreateIndex
|
||||||
|
CREATE INDEX "works_author_idx" ON "works"("author");
|
||||||
|
|
||||||
|
-- CreateIndex
|
||||||
|
CREATE UNIQUE INDEX "work_asins_asin_key" ON "work_asins"("asin");
|
||||||
|
|
||||||
|
-- CreateIndex
|
||||||
|
CREATE INDEX "work_asins_work_id_idx" ON "work_asins"("work_id");
|
||||||
|
|
||||||
|
-- CreateIndex
|
||||||
|
CREATE INDEX "work_asins_asin_idx" ON "work_asins"("asin");
|
||||||
|
|
||||||
|
-- AddForeignKey
|
||||||
|
ALTER TABLE "work_asins" ADD CONSTRAINT "work_asins_work_id_fkey" FOREIGN KEY ("work_id") REFERENCES "works"("id") ON DELETE CASCADE ON UPDATE CASCADE;
|
||||||
@@ -0,0 +1,51 @@
|
|||||||
|
-- CreateTable
|
||||||
|
CREATE TABLE "watched_series" (
|
||||||
|
"id" TEXT NOT NULL,
|
||||||
|
"user_id" TEXT NOT NULL,
|
||||||
|
"series_asin" TEXT NOT NULL,
|
||||||
|
"series_title" TEXT NOT NULL,
|
||||||
|
"cover_art_url" TEXT,
|
||||||
|
"last_checked_at" TIMESTAMP(3),
|
||||||
|
"created_at" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
|
||||||
|
"updated_at" TIMESTAMP(3) NOT NULL,
|
||||||
|
|
||||||
|
CONSTRAINT "watched_series_pkey" PRIMARY KEY ("id")
|
||||||
|
);
|
||||||
|
|
||||||
|
-- CreateTable
|
||||||
|
CREATE TABLE "watched_authors" (
|
||||||
|
"id" TEXT NOT NULL,
|
||||||
|
"user_id" TEXT NOT NULL,
|
||||||
|
"author_asin" TEXT NOT NULL,
|
||||||
|
"author_name" TEXT NOT NULL,
|
||||||
|
"cover_art_url" TEXT,
|
||||||
|
"last_checked_at" TIMESTAMP(3),
|
||||||
|
"created_at" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
|
||||||
|
"updated_at" TIMESTAMP(3) NOT NULL,
|
||||||
|
|
||||||
|
CONSTRAINT "watched_authors_pkey" PRIMARY KEY ("id")
|
||||||
|
);
|
||||||
|
|
||||||
|
-- CreateIndex
|
||||||
|
CREATE INDEX "watched_series_user_id_idx" ON "watched_series"("user_id");
|
||||||
|
|
||||||
|
-- CreateIndex
|
||||||
|
CREATE INDEX "watched_series_series_asin_idx" ON "watched_series"("series_asin");
|
||||||
|
|
||||||
|
-- CreateIndex
|
||||||
|
CREATE UNIQUE INDEX "watched_series_user_id_series_asin_key" ON "watched_series"("user_id", "series_asin");
|
||||||
|
|
||||||
|
-- CreateIndex
|
||||||
|
CREATE INDEX "watched_authors_user_id_idx" ON "watched_authors"("user_id");
|
||||||
|
|
||||||
|
-- CreateIndex
|
||||||
|
CREATE INDEX "watched_authors_author_asin_idx" ON "watched_authors"("author_asin");
|
||||||
|
|
||||||
|
-- CreateIndex
|
||||||
|
CREATE UNIQUE INDEX "watched_authors_user_id_author_asin_key" ON "watched_authors"("user_id", "author_asin");
|
||||||
|
|
||||||
|
-- AddForeignKey
|
||||||
|
ALTER TABLE "watched_series" ADD CONSTRAINT "watched_series_user_id_fkey" FOREIGN KEY ("user_id") REFERENCES "users"("id") ON DELETE CASCADE ON UPDATE CASCADE;
|
||||||
|
|
||||||
|
-- AddForeignKey
|
||||||
|
ALTER TABLE "watched_authors" ADD CONSTRAINT "watched_authors_user_id_fkey" FOREIGN KEY ("user_id") REFERENCES "users"("id") ON DELETE CASCADE ON UPDATE CASCADE;
|
||||||
@@ -0,0 +1,49 @@
|
|||||||
|
-- CreateTable
|
||||||
|
CREATE TABLE "hardcover_shelves" (
|
||||||
|
"id" TEXT NOT NULL,
|
||||||
|
"user_id" TEXT NOT NULL,
|
||||||
|
"name" TEXT NOT NULL,
|
||||||
|
"list_id" TEXT NOT NULL,
|
||||||
|
"api_token" TEXT NOT NULL,
|
||||||
|
"last_sync_at" TIMESTAMP(3),
|
||||||
|
"book_count" INTEGER,
|
||||||
|
"cover_urls" TEXT,
|
||||||
|
"created_at" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
|
||||||
|
"updated_at" TIMESTAMP(3) NOT NULL,
|
||||||
|
|
||||||
|
CONSTRAINT "hardcover_shelves_pkey" PRIMARY KEY ("id")
|
||||||
|
);
|
||||||
|
|
||||||
|
-- CreateTable
|
||||||
|
CREATE TABLE "hardcover_book_mappings" (
|
||||||
|
"id" TEXT NOT NULL,
|
||||||
|
"hardcover_book_id" TEXT NOT NULL,
|
||||||
|
"title" TEXT NOT NULL,
|
||||||
|
"author" TEXT NOT NULL,
|
||||||
|
"audible_asin" TEXT,
|
||||||
|
"cover_url" TEXT,
|
||||||
|
"no_match" BOOLEAN NOT NULL DEFAULT false,
|
||||||
|
"last_search_at" TIMESTAMP(3),
|
||||||
|
"created_at" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
|
||||||
|
"updated_at" TIMESTAMP(3) NOT NULL,
|
||||||
|
|
||||||
|
CONSTRAINT "hardcover_book_mappings_pkey" PRIMARY KEY ("id")
|
||||||
|
);
|
||||||
|
|
||||||
|
-- CreateIndex
|
||||||
|
CREATE INDEX "hardcover_shelves_user_id_idx" ON "hardcover_shelves"("user_id");
|
||||||
|
|
||||||
|
-- CreateIndex
|
||||||
|
CREATE UNIQUE INDEX "hardcover_shelves_user_id_list_id_key" ON "hardcover_shelves"("user_id", "list_id");
|
||||||
|
|
||||||
|
-- CreateIndex
|
||||||
|
CREATE UNIQUE INDEX "hardcover_book_mappings_hardcover_book_id_key" ON "hardcover_book_mappings"("hardcover_book_id");
|
||||||
|
|
||||||
|
-- CreateIndex
|
||||||
|
CREATE INDEX "hardcover_book_mappings_hardcover_book_id_idx" ON "hardcover_book_mappings"("hardcover_book_id");
|
||||||
|
|
||||||
|
-- CreateIndex
|
||||||
|
CREATE INDEX "hardcover_book_mappings_audible_asin_idx" ON "hardcover_book_mappings"("audible_asin");
|
||||||
|
|
||||||
|
-- AddForeignKey
|
||||||
|
ALTER TABLE "hardcover_shelves" ADD CONSTRAINT "hardcover_shelves_user_id_fkey" FOREIGN KEY ("user_id") REFERENCES "users"("id") ON DELETE CASCADE ON UPDATE CASCADE;
|
||||||
@@ -0,0 +1,3 @@
|
|||||||
|
-- Normalize existing local usernames to lowercase
|
||||||
|
UPDATE users SET plex_username = LOWER(plex_username) WHERE auth_provider = 'local' AND deleted_at IS NULL;
|
||||||
|
UPDATE users SET plex_id = 'local-' || LOWER(SUBSTRING(plex_id FROM 7)) WHERE plex_id LIKE 'local-%' AND plex_id NOT LIKE 'local-%-deleted-%';
|
||||||
@@ -0,0 +1,41 @@
|
|||||||
|
-- CreateTable
|
||||||
|
CREATE TABLE "book_mappings" (
|
||||||
|
"id" TEXT NOT NULL,
|
||||||
|
"provider" TEXT NOT NULL,
|
||||||
|
"external_book_id" TEXT NOT NULL,
|
||||||
|
"title" TEXT NOT NULL,
|
||||||
|
"author" TEXT NOT NULL,
|
||||||
|
"audible_asin" TEXT,
|
||||||
|
"cover_url" TEXT,
|
||||||
|
"no_match" BOOLEAN NOT NULL DEFAULT false,
|
||||||
|
"last_search_at" TIMESTAMP(3),
|
||||||
|
"created_at" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
|
||||||
|
"updated_at" TIMESTAMP(3) NOT NULL,
|
||||||
|
|
||||||
|
CONSTRAINT "book_mappings_pkey" PRIMARY KEY ("id")
|
||||||
|
);
|
||||||
|
|
||||||
|
-- Migrate data from goodreads_book_mappings
|
||||||
|
INSERT INTO "book_mappings" ("id", "provider", "external_book_id", "title", "author", "audible_asin", "cover_url", "no_match", "last_search_at", "created_at", "updated_at")
|
||||||
|
SELECT "id", 'goodreads', "goodreads_book_id", "title", "author", "audible_asin", "cover_url", "no_match", "last_search_at", "created_at", "updated_at"
|
||||||
|
FROM "goodreads_book_mappings";
|
||||||
|
|
||||||
|
-- Migrate data from hardcover_book_mappings
|
||||||
|
INSERT INTO "book_mappings" ("id", "provider", "external_book_id", "title", "author", "audible_asin", "cover_url", "no_match", "last_search_at", "created_at", "updated_at")
|
||||||
|
SELECT "id", 'hardcover', "hardcover_book_id", "title", "author", "audible_asin", "cover_url", "no_match", "last_search_at", "created_at", "updated_at"
|
||||||
|
FROM "hardcover_book_mappings";
|
||||||
|
|
||||||
|
-- DropTable
|
||||||
|
DROP TABLE "goodreads_book_mappings";
|
||||||
|
|
||||||
|
-- DropTable
|
||||||
|
DROP TABLE "hardcover_book_mappings";
|
||||||
|
|
||||||
|
-- CreateIndex
|
||||||
|
CREATE UNIQUE INDEX "book_mappings_provider_external_book_id_key" ON "book_mappings"("provider", "external_book_id");
|
||||||
|
|
||||||
|
-- CreateIndex
|
||||||
|
CREATE INDEX "book_mappings_provider_external_book_id_idx" ON "book_mappings"("provider", "external_book_id");
|
||||||
|
|
||||||
|
-- CreateIndex
|
||||||
|
CREATE INDEX "book_mappings_audible_asin_idx" ON "book_mappings"("audible_asin");
|
||||||
@@ -0,0 +1,33 @@
|
|||||||
|
-- CreateTable
|
||||||
|
CREATE TABLE "api_tokens" (
|
||||||
|
"id" TEXT NOT NULL,
|
||||||
|
"name" TEXT NOT NULL,
|
||||||
|
"token_hash" TEXT NOT NULL,
|
||||||
|
"token_prefix" TEXT NOT NULL,
|
||||||
|
"role" TEXT NOT NULL DEFAULT 'user',
|
||||||
|
"created_by_id" TEXT NOT NULL,
|
||||||
|
"user_id" TEXT NOT NULL,
|
||||||
|
"last_used_at" TIMESTAMP(3),
|
||||||
|
"expires_at" TIMESTAMP(3),
|
||||||
|
"created_at" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
|
||||||
|
|
||||||
|
CONSTRAINT "api_tokens_pkey" PRIMARY KEY ("id")
|
||||||
|
);
|
||||||
|
|
||||||
|
-- CreateIndex
|
||||||
|
CREATE UNIQUE INDEX "api_tokens_token_hash_key" ON "api_tokens"("token_hash");
|
||||||
|
|
||||||
|
-- CreateIndex
|
||||||
|
CREATE INDEX "api_tokens_token_hash_idx" ON "api_tokens"("token_hash");
|
||||||
|
|
||||||
|
-- CreateIndex
|
||||||
|
CREATE INDEX "api_tokens_created_by_id_idx" ON "api_tokens"("created_by_id");
|
||||||
|
|
||||||
|
-- CreateIndex
|
||||||
|
CREATE INDEX "api_tokens_user_id_idx" ON "api_tokens"("user_id");
|
||||||
|
|
||||||
|
-- AddForeignKey
|
||||||
|
ALTER TABLE "api_tokens" ADD CONSTRAINT "api_tokens_created_by_id_fkey" FOREIGN KEY ("created_by_id") REFERENCES "users"("id") ON DELETE CASCADE ON UPDATE CASCADE;
|
||||||
|
|
||||||
|
-- AddForeignKey
|
||||||
|
ALTER TABLE "api_tokens" ADD CONSTRAINT "api_tokens_user_id_fkey" FOREIGN KEY ("user_id") REFERENCES "users"("id") ON DELETE CASCADE ON UPDATE CASCADE;
|
||||||
@@ -0,0 +1,49 @@
|
|||||||
|
-- CreateTable
|
||||||
|
CREATE TABLE "user_home_sections" (
|
||||||
|
"id" TEXT NOT NULL,
|
||||||
|
"user_id" TEXT NOT NULL,
|
||||||
|
"section_type" TEXT NOT NULL,
|
||||||
|
"category_id" TEXT,
|
||||||
|
"category_name" TEXT,
|
||||||
|
"sort_order" INTEGER NOT NULL,
|
||||||
|
"created_at" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
|
||||||
|
"updated_at" TIMESTAMP(3) NOT NULL,
|
||||||
|
|
||||||
|
CONSTRAINT "user_home_sections_pkey" PRIMARY KEY ("id")
|
||||||
|
);
|
||||||
|
|
||||||
|
-- CreateTable
|
||||||
|
CREATE TABLE "audible_cache_categories" (
|
||||||
|
"id" TEXT NOT NULL,
|
||||||
|
"asin" TEXT NOT NULL,
|
||||||
|
"category_id" TEXT NOT NULL,
|
||||||
|
"rank" INTEGER NOT NULL,
|
||||||
|
"last_synced_at" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
|
||||||
|
"created_at" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
|
||||||
|
|
||||||
|
CONSTRAINT "audible_cache_categories_pkey" PRIMARY KEY ("id")
|
||||||
|
);
|
||||||
|
|
||||||
|
-- CreateIndex
|
||||||
|
CREATE INDEX "user_home_sections_user_id_idx" ON "user_home_sections"("user_id");
|
||||||
|
|
||||||
|
-- CreateIndex
|
||||||
|
CREATE INDEX "user_home_sections_sort_order_idx" ON "user_home_sections"("sort_order");
|
||||||
|
|
||||||
|
-- CreateIndex
|
||||||
|
CREATE UNIQUE INDEX "user_home_sections_user_id_section_type_category_id_key" ON "user_home_sections"("user_id", "section_type", "category_id");
|
||||||
|
|
||||||
|
-- CreateIndex
|
||||||
|
CREATE INDEX "audible_cache_categories_category_id_idx" ON "audible_cache_categories"("category_id");
|
||||||
|
|
||||||
|
-- CreateIndex
|
||||||
|
CREATE INDEX "audible_cache_categories_asin_idx" ON "audible_cache_categories"("asin");
|
||||||
|
|
||||||
|
-- CreateIndex
|
||||||
|
CREATE INDEX "audible_cache_categories_category_id_rank_idx" ON "audible_cache_categories"("category_id", "rank");
|
||||||
|
|
||||||
|
-- CreateIndex
|
||||||
|
CREATE UNIQUE INDEX "audible_cache_categories_asin_category_id_key" ON "audible_cache_categories"("asin", "category_id");
|
||||||
|
|
||||||
|
-- AddForeignKey
|
||||||
|
ALTER TABLE "user_home_sections" ADD CONSTRAINT "user_home_sections_user_id_fkey" FOREIGN KEY ("user_id") REFERENCES "users"("id") ON DELETE CASCADE ON UPDATE CASCADE;
|
||||||
@@ -0,0 +1,17 @@
|
|||||||
|
-- DropIndex
|
||||||
|
DROP INDEX IF EXISTS "audible_cache_is_popular_idx";
|
||||||
|
|
||||||
|
-- DropIndex
|
||||||
|
DROP INDEX IF EXISTS "audible_cache_is_new_release_idx";
|
||||||
|
|
||||||
|
-- DropIndex
|
||||||
|
DROP INDEX IF EXISTS "audible_cache_popular_rank_idx";
|
||||||
|
|
||||||
|
-- DropIndex
|
||||||
|
DROP INDEX IF EXISTS "audible_cache_new_release_rank_idx";
|
||||||
|
|
||||||
|
-- AlterTable - Remove legacy discovery flag columns (now stored in audible_cache_categories)
|
||||||
|
ALTER TABLE "audible_cache" DROP COLUMN "is_popular",
|
||||||
|
DROP COLUMN "is_new_release",
|
||||||
|
DROP COLUMN "popular_rank",
|
||||||
|
DROP COLUMN "new_release_rank";
|
||||||
+213
-24
@@ -55,6 +55,7 @@ model User {
|
|||||||
|
|
||||||
// Fine-grained permissions
|
// Fine-grained permissions
|
||||||
interactiveSearchAccess Boolean? @map("interactive_search_access") // null = use global setting, true = allow, false = deny
|
interactiveSearchAccess Boolean? @map("interactive_search_access") // null = use global setting, true = allow, false = deny
|
||||||
|
downloadAccess Boolean? @map("download_access") // null = use global setting, true = allow, false = deny
|
||||||
|
|
||||||
// Soft delete support
|
// Soft delete support
|
||||||
deletedAt DateTime? @map("deleted_at")
|
deletedAt DateTime? @map("deleted_at")
|
||||||
@@ -65,8 +66,14 @@ model User {
|
|||||||
bookDateRecommendations BookDateRecommendation[]
|
bookDateRecommendations BookDateRecommendation[]
|
||||||
bookDateSwipes BookDateSwipe[]
|
bookDateSwipes BookDateSwipe[]
|
||||||
goodreadsShelves GoodreadsShelf[]
|
goodreadsShelves GoodreadsShelf[]
|
||||||
|
hardcoverShelves HardcoverShelf[]
|
||||||
reportedIssues ReportedIssue[] @relation("Reporter")
|
reportedIssues ReportedIssue[] @relation("Reporter")
|
||||||
resolvedIssues ReportedIssue[] @relation("Resolver")
|
resolvedIssues ReportedIssue[] @relation("Resolver")
|
||||||
|
createdApiTokens ApiToken[] @relation("CreatedApiTokens")
|
||||||
|
apiTokens ApiToken[] @relation("UserApiTokens")
|
||||||
|
watchedSeries WatchedSeries[]
|
||||||
|
watchedAuthors WatchedAuthor[]
|
||||||
|
homeSections UserHomeSection[]
|
||||||
|
|
||||||
@@index([plexId])
|
@@index([plexId])
|
||||||
@@index([role])
|
@@index([role])
|
||||||
@@ -93,12 +100,6 @@ model AudibleCache {
|
|||||||
rating Decimal? @db.Decimal(3, 2)
|
rating Decimal? @db.Decimal(3, 2)
|
||||||
genres Json @default("[]")
|
genres Json @default("[]")
|
||||||
|
|
||||||
// Discovery categories
|
|
||||||
isPopular Boolean @default(false) @map("is_popular")
|
|
||||||
isNewRelease Boolean @default(false) @map("is_new_release")
|
|
||||||
popularRank Int? @map("popular_rank")
|
|
||||||
newReleaseRank Int? @map("new_release_rank")
|
|
||||||
|
|
||||||
lastSyncedAt DateTime @default(now()) @map("last_synced_at")
|
lastSyncedAt DateTime @default(now()) @map("last_synced_at")
|
||||||
createdAt DateTime @default(now()) @map("created_at")
|
createdAt DateTime @default(now()) @map("created_at")
|
||||||
updatedAt DateTime @updatedAt @map("updated_at")
|
updatedAt DateTime @updatedAt @map("updated_at")
|
||||||
@@ -106,10 +107,6 @@ model AudibleCache {
|
|||||||
@@index([asin])
|
@@index([asin])
|
||||||
@@index([title])
|
@@index([title])
|
||||||
@@index([author])
|
@@index([author])
|
||||||
@@index([isPopular])
|
|
||||||
@@index([isNewRelease])
|
|
||||||
@@index([popularRank])
|
|
||||||
@@index([newReleaseRank])
|
|
||||||
@@map("audible_cache")
|
@@map("audible_cache")
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -231,6 +228,7 @@ model Request {
|
|||||||
importAttempts Int @default(0) @map("import_attempts")
|
importAttempts Int @default(0) @map("import_attempts")
|
||||||
maxImportRetries Int @default(5) @map("max_import_retries")
|
maxImportRetries Int @default(5) @map("max_import_retries")
|
||||||
lastSearchAt DateTime? @map("last_search_at")
|
lastSearchAt DateTime? @map("last_search_at")
|
||||||
|
customSearchTerms String? @map("custom_search_terms") @db.Text
|
||||||
lastImportAt DateTime? @map("last_import_at")
|
lastImportAt DateTime? @map("last_import_at")
|
||||||
createdAt DateTime @default(now()) @map("created_at")
|
createdAt DateTime @default(now()) @map("created_at")
|
||||||
updatedAt DateTime @updatedAt @map("updated_at")
|
updatedAt DateTime @updatedAt @map("updated_at")
|
||||||
@@ -390,7 +388,7 @@ model ScheduledJob {
|
|||||||
|
|
||||||
model BookDateConfig {
|
model BookDateConfig {
|
||||||
id String @id @default(uuid())
|
id String @id @default(uuid())
|
||||||
provider String // 'openai' | 'claude' | 'custom'
|
provider String // 'openai' | 'claude' | 'gemini' | 'custom'
|
||||||
apiKey String @map("api_key") @db.Text // Encrypted at rest (AES-256)
|
apiKey String @map("api_key") @db.Text // Encrypted at rest (AES-256)
|
||||||
model String // e.g., 'gpt-4o', 'claude-sonnet-4-5-20250929'
|
model String // e.g., 'gpt-4o', 'claude-sonnet-4-5-20250929'
|
||||||
baseUrl String? @map("base_url") @db.Text // Base URL for custom provider (OpenAI-compatible endpoints)
|
baseUrl String? @map("base_url") @db.Text // Base URL for custom provider (OpenAI-compatible endpoints)
|
||||||
@@ -494,6 +492,34 @@ model ReportedIssue {
|
|||||||
// Per-user Goodreads shelf subscriptions + global book-to-ASIN mapping cache
|
// Per-user Goodreads shelf subscriptions + global book-to-ASIN mapping cache
|
||||||
// ============================================================================
|
// ============================================================================
|
||||||
|
|
||||||
|
// ============================================================================
|
||||||
|
// API TOKEN TABLE
|
||||||
|
// Static API tokens for programmatic access (alternative to JWT)
|
||||||
|
// Documentation: documentation/backend/services/api-tokens.md
|
||||||
|
// ============================================================================
|
||||||
|
|
||||||
|
model ApiToken {
|
||||||
|
id String @id @default(uuid())
|
||||||
|
name String // User-friendly label (e.g., "Home Assistant", "Webhook")
|
||||||
|
tokenHash String @unique @map("token_hash") // SHA-256 hash of the token (never store plaintext)
|
||||||
|
tokenPrefix String @map("token_prefix") // First 8 chars for display (e.g., "rmab_a1b2")
|
||||||
|
role String @default("user") // Token role: 'admin' or 'user'
|
||||||
|
createdById String @map("created_by_id") // Who created the token (may differ from userId for admin-created tokens)
|
||||||
|
userId String @map("user_id") // The user identity this token acts as
|
||||||
|
lastUsedAt DateTime? @map("last_used_at")
|
||||||
|
expiresAt DateTime? @map("expires_at") // null = never expires
|
||||||
|
createdAt DateTime @default(now()) @map("created_at")
|
||||||
|
|
||||||
|
// Relations
|
||||||
|
createdBy User @relation("CreatedApiTokens", fields: [createdById], references: [id], onDelete: Cascade)
|
||||||
|
tokenUser User @relation("UserApiTokens", fields: [userId], references: [id], onDelete: Cascade)
|
||||||
|
|
||||||
|
@@index([tokenHash])
|
||||||
|
@@index([createdById])
|
||||||
|
@@index([userId])
|
||||||
|
@@map("api_tokens")
|
||||||
|
}
|
||||||
|
|
||||||
model GoodreadsShelf {
|
model GoodreadsShelf {
|
||||||
id String @id @default(uuid())
|
id String @id @default(uuid())
|
||||||
userId String @map("user_id")
|
userId String @map("user_id")
|
||||||
@@ -513,19 +539,182 @@ model GoodreadsShelf {
|
|||||||
@@map("goodreads_shelves")
|
@@map("goodreads_shelves")
|
||||||
}
|
}
|
||||||
|
|
||||||
model GoodreadsBookMapping {
|
// ============================================================================
|
||||||
id String @id @default(uuid())
|
// UNIFIED BOOK MAPPING TABLE
|
||||||
goodreadsBookId String @unique @map("goodreads_book_id")
|
// Global book-to-ASIN mapping cache shared across all shelf providers.
|
||||||
title String
|
// Uses provider + externalBookId composite key for cross-provider dedup.
|
||||||
author String
|
// ============================================================================
|
||||||
audibleAsin String? @map("audible_asin")
|
|
||||||
coverUrl String? @map("cover_url") @db.Text
|
|
||||||
noMatch Boolean @default(false) @map("no_match")
|
|
||||||
lastSearchAt DateTime? @map("last_search_at")
|
|
||||||
createdAt DateTime @default(now()) @map("created_at")
|
|
||||||
updatedAt DateTime @updatedAt @map("updated_at")
|
|
||||||
|
|
||||||
@@index([goodreadsBookId])
|
model BookMapping {
|
||||||
|
id String @id @default(uuid())
|
||||||
|
provider String // "goodreads", "hardcover", etc.
|
||||||
|
externalBookId String @map("external_book_id")
|
||||||
|
title String
|
||||||
|
author String
|
||||||
|
audibleAsin String? @map("audible_asin")
|
||||||
|
coverUrl String? @map("cover_url") @db.Text
|
||||||
|
noMatch Boolean @default(false) @map("no_match")
|
||||||
|
lastSearchAt DateTime? @map("last_search_at")
|
||||||
|
createdAt DateTime @default(now()) @map("created_at")
|
||||||
|
updatedAt DateTime @updatedAt @map("updated_at")
|
||||||
|
|
||||||
|
@@unique([provider, externalBookId])
|
||||||
|
@@index([provider, externalBookId])
|
||||||
@@index([audibleAsin])
|
@@index([audibleAsin])
|
||||||
@@map("goodreads_book_mappings")
|
@@map("book_mappings")
|
||||||
|
}
|
||||||
|
|
||||||
|
// ============================================================================
|
||||||
|
// HARDCOVER SYNC TABLES
|
||||||
|
// Per-user Hardcover list subscriptions
|
||||||
|
// ============================================================================
|
||||||
|
|
||||||
|
model HardcoverShelf {
|
||||||
|
id String @id @default(uuid())
|
||||||
|
userId String @map("user_id")
|
||||||
|
name String // Extracted from Hardcover API list name or status
|
||||||
|
listId String @map("list_id") // Hardcover List ID or Status ID
|
||||||
|
apiToken String @map("api_token") @db.Text // User's personal access token for hardcover api
|
||||||
|
lastSyncAt DateTime? @map("last_sync_at")
|
||||||
|
bookCount Int? @map("book_count")
|
||||||
|
coverUrls String? @map("cover_urls") @db.Text // JSON array of cover image URLs
|
||||||
|
createdAt DateTime @default(now()) @map("created_at")
|
||||||
|
updatedAt DateTime @updatedAt @map("updated_at")
|
||||||
|
|
||||||
|
// Relations
|
||||||
|
user User @relation(fields: [userId], references: [id], onDelete: Cascade)
|
||||||
|
|
||||||
|
@@unique([userId, listId])
|
||||||
|
@@index([userId])
|
||||||
|
@@map("hardcover_shelves")
|
||||||
|
}
|
||||||
|
|
||||||
|
// ============================================================================
|
||||||
|
// WORKS TABLE
|
||||||
|
// Cross-ASIN audiobook identity mapping — links multiple Audible ASINs
|
||||||
|
// to a single logical work for library matching across editions.
|
||||||
|
// Documentation: documentation/integrations/audible.md
|
||||||
|
// ============================================================================
|
||||||
|
|
||||||
|
model Work {
|
||||||
|
id String @id @default(uuid())
|
||||||
|
title String
|
||||||
|
author String
|
||||||
|
createdAt DateTime @default(now()) @map("created_at")
|
||||||
|
updatedAt DateTime @updatedAt @map("updated_at")
|
||||||
|
|
||||||
|
// Relations
|
||||||
|
asins WorkAsin[]
|
||||||
|
|
||||||
|
@@index([title])
|
||||||
|
@@index([author])
|
||||||
|
@@map("works")
|
||||||
|
}
|
||||||
|
|
||||||
|
model WorkAsin {
|
||||||
|
id String @id @default(uuid())
|
||||||
|
workId String @map("work_id")
|
||||||
|
asin String @unique
|
||||||
|
narrator String?
|
||||||
|
durationMinutes Int? @map("duration_minutes")
|
||||||
|
isCanonical Boolean @default(false) @map("is_canonical")
|
||||||
|
source String // 'dedup_auto' | 'admin_manual'
|
||||||
|
createdAt DateTime @default(now()) @map("created_at")
|
||||||
|
|
||||||
|
// Relations
|
||||||
|
work Work @relation(fields: [workId], references: [id], onDelete: Cascade)
|
||||||
|
|
||||||
|
@@index([workId])
|
||||||
|
@@index([asin])
|
||||||
|
@@map("work_asins")
|
||||||
|
}
|
||||||
|
|
||||||
|
// ============================================================================
|
||||||
|
// WATCHED LISTS TABLES
|
||||||
|
// Per-user series and author subscriptions for automatic new-release requests.
|
||||||
|
// Documentation: documentation/features/watched-lists.md
|
||||||
|
// ============================================================================
|
||||||
|
|
||||||
|
model WatchedSeries {
|
||||||
|
id String @id @default(uuid())
|
||||||
|
userId String @map("user_id")
|
||||||
|
seriesAsin String @map("series_asin")
|
||||||
|
seriesTitle String @map("series_title")
|
||||||
|
coverArtUrl String? @map("cover_art_url") @db.Text
|
||||||
|
lastCheckedAt DateTime? @map("last_checked_at")
|
||||||
|
createdAt DateTime @default(now()) @map("created_at")
|
||||||
|
updatedAt DateTime @updatedAt @map("updated_at")
|
||||||
|
|
||||||
|
// Relations
|
||||||
|
user User @relation(fields: [userId], references: [id], onDelete: Cascade)
|
||||||
|
|
||||||
|
@@unique([userId, seriesAsin])
|
||||||
|
@@index([userId])
|
||||||
|
@@index([seriesAsin])
|
||||||
|
@@map("watched_series")
|
||||||
|
}
|
||||||
|
|
||||||
|
model WatchedAuthor {
|
||||||
|
id String @id @default(uuid())
|
||||||
|
userId String @map("user_id")
|
||||||
|
authorAsin String @map("author_asin")
|
||||||
|
authorName String @map("author_name")
|
||||||
|
coverArtUrl String? @map("cover_art_url") @db.Text
|
||||||
|
lastCheckedAt DateTime? @map("last_checked_at")
|
||||||
|
createdAt DateTime @default(now()) @map("created_at")
|
||||||
|
updatedAt DateTime @updatedAt @map("updated_at")
|
||||||
|
|
||||||
|
// Relations
|
||||||
|
user User @relation(fields: [userId], references: [id], onDelete: Cascade)
|
||||||
|
|
||||||
|
@@unique([userId, authorAsin])
|
||||||
|
@@index([userId])
|
||||||
|
@@index([authorAsin])
|
||||||
|
@@map("watched_authors")
|
||||||
|
}
|
||||||
|
|
||||||
|
// ============================================================================
|
||||||
|
// USER HOME SECTION TABLE
|
||||||
|
// Per-user configurable home page sections (popular, new_releases, category)
|
||||||
|
// Documentation: documentation/features/home-sections.md
|
||||||
|
// ============================================================================
|
||||||
|
|
||||||
|
model UserHomeSection {
|
||||||
|
id String @id @default(uuid())
|
||||||
|
userId String @map("user_id")
|
||||||
|
sectionType String @map("section_type") // 'popular' | 'new_releases' | 'category'
|
||||||
|
categoryId String? @map("category_id") // Audible category node ID (only for type 'category')
|
||||||
|
categoryName String? @map("category_name") // Display name (only for type 'category')
|
||||||
|
sortOrder Int @map("sort_order")
|
||||||
|
createdAt DateTime @default(now()) @map("created_at")
|
||||||
|
updatedAt DateTime @updatedAt @map("updated_at")
|
||||||
|
|
||||||
|
// Relations
|
||||||
|
user User @relation(fields: [userId], references: [id], onDelete: Cascade)
|
||||||
|
|
||||||
|
@@unique([userId, sectionType, categoryId])
|
||||||
|
@@index([userId])
|
||||||
|
@@index([sortOrder])
|
||||||
|
@@map("user_home_sections")
|
||||||
|
}
|
||||||
|
|
||||||
|
// ============================================================================
|
||||||
|
// AUDIBLE CACHE CATEGORY TABLE
|
||||||
|
// Join table linking AudibleCache entries to Audible categories with ranking
|
||||||
|
// Documentation: documentation/features/home-sections.md
|
||||||
|
// ============================================================================
|
||||||
|
|
||||||
|
model AudibleCacheCategory {
|
||||||
|
id String @id @default(uuid())
|
||||||
|
asin String
|
||||||
|
categoryId String @map("category_id")
|
||||||
|
rank Int
|
||||||
|
lastSyncedAt DateTime @default(now()) @map("last_synced_at")
|
||||||
|
createdAt DateTime @default(now()) @map("created_at")
|
||||||
|
|
||||||
|
@@unique([asin, categoryId])
|
||||||
|
@@index([categoryId])
|
||||||
|
@@index([asin])
|
||||||
|
@@index([categoryId, rank])
|
||||||
|
@@map("audible_cache_categories")
|
||||||
}
|
}
|
||||||
|
|||||||
Binary file not shown.
|
After Width: | Height: | Size: 1.0 KiB |
@@ -0,0 +1 @@
|
|||||||
|
<svg xmlns="http://www.w3.org/2000/svg" class="w-9 group-hover:rotate-12 transition-all duration-300" fill="none" viewBox="0 0 40 40"><path d="M12.8889 32.5982C12.666 31.7661 13.1598 30.9108 13.9919 30.6879L30.2971 26.3189C31.1292 26.096 31.9845 26.5898 32.2075 27.4219L32.8739 29.9089C33.1711 31.0183 32.5127 32.1587 31.4033 32.456L18.1113 36.0176C15.8924 36.6121 13.6116 35.2953 13.0171 33.0764L12.8889 32.5982Z" fill="#4F46E5"></path><path d="M7.62314 12.946C7.05137 10.8121 8.3177 8.61876 10.4516 8.04699L16.8851 32.0571L13.0214 33.0924L7.62314 12.946Z" fill="#4F46E5"></path><path d="M29.3358 24.432L31.2677 23.9144L32.3584 27.985C32.6443 29.052 32.0111 30.1486 30.9442 30.4345L29.3358 24.432Z" fill="#4338CA"></path><path d="M26.4446 5.91475C26.1474 4.80529 25.007 4.14688 23.8975 4.44416L10.5286 8.02636C9.41911 8.32364 8.7607 9.46403 9.05798 10.5735L14.9532 32.5748L22.6461 30.5135C23.1986 30.3654 23.5265 29.7975 23.3785 29.245C23.2304 28.6925 23.5583 28.1245 24.1108 27.9765L29.7949 26.4535C30.9043 26.1562 31.5628 25.0158 31.2655 23.9063L26.4446 5.91475Z" fill="#6366F1"></path><path d="M21.0947 11.2811C21.145 10.6645 21.9408 10.4512 22.2927 10.9601L22.442 11.1761C22.5512 11.3341 22.724 11.4365 22.9151 11.4565L23.2375 11.4902C23.838 11.553 24.0445 12.3235 23.5558 12.6781L23.2935 12.8685C23.138 12.9813 23.0395 13.1564 23.0239 13.3479L23.0026 13.6096C22.9523 14.2262 22.1564 14.4394 21.8046 13.9306L21.6553 13.7146C21.546 13.5566 21.3732 13.4542 21.1821 13.4342L20.8598 13.4005C20.2592 13.3377 20.0528 12.5672 20.5415 12.2126L20.8038 12.0222C20.9593 11.9094 21.0577 11.7343 21.0734 11.5428L21.0947 11.2811Z" fill="#312E81"></path><path d="M18.3031 16.3181C18.3533 15.7015 19.1492 15.4882 19.501 15.9971L20.5634 17.5337C20.6727 17.6917 20.8455 17.7941 21.0366 17.8141L22.9139 18.0104C23.5144 18.0732 23.7208 18.8436 23.2321 19.1983L21.7045 20.3069C21.549 20.4197 21.4506 20.5949 21.435 20.7863L21.2832 22.6482C21.2329 23.2649 20.4371 23.4781 20.0852 22.9692L19.0228 21.4327C18.9136 21.2747 18.7407 21.1722 18.5497 21.1522L16.6724 20.956C16.0719 20.8932 15.8654 20.1227 16.3541 19.7681L17.8817 18.6594C18.0372 18.5466 18.1357 18.3715 18.1513 18.18L18.3031 16.3181Z" fill="#312E81"></path><path d="M14.9532 32.5748C14.6571 31.4697 15.3129 30.3339 16.4179 30.0378L29.8719 26.4328L30.9441 30.4345L17.4902 34.0395C16.3851 34.3356 15.2493 33.6798 14.9532 32.5748Z" fill="#EEF2FF"></path></svg>
|
||||||
|
After Width: | Height: | Size: 2.3 KiB |
@@ -0,0 +1,22 @@
|
|||||||
|
<?xml version="1.0" encoding="UTF-8" standalone="no"?>
|
||||||
|
<svg width="500px" height="500px" viewBox="0 0 500 500" version="1.1" xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink">
|
||||||
|
<!-- Generator: Sketch 41.2 (35397) - http://www.bohemiancoding.com/sketch -->
|
||||||
|
<title>img-coverart</title>
|
||||||
|
<desc>Created with Sketch.</desc>
|
||||||
|
<defs>
|
||||||
|
<rect id="path-1" x="0" y="0" width="500" height="500"></rect>
|
||||||
|
</defs>
|
||||||
|
<g id="Page-1" stroke="none" stroke-width="1" fill="none" fill-rule="evenodd">
|
||||||
|
<g id="Account-details:-membership-asin-doc" transform="translate(-87.000000, -867.000000)">
|
||||||
|
<g id="Group" transform="translate(65.000000, 780.000000)">
|
||||||
|
<g id="img-coverart" transform="translate(22.000000, 87.000000)">
|
||||||
|
<mask id="mask-2" fill="white">
|
||||||
|
<use xlink:href="#path-1"></use>
|
||||||
|
</mask>
|
||||||
|
<use id="mask" fill="#BBBBBB" xlink:href="#path-1"></use>
|
||||||
|
<path d="M251.314605,307.191176 L126.315789,229.090627 L126.315789,250.186562 L251.314605,328.289474 L376.315789,250.186562 L376.315789,229.090627 L251.314605,307.191176 Z M300.338486,257.198698 L318.743522,245.697622 L318.757695,245.697622 C304.238718,223.902504 279.436447,209.540923 251.277279,209.540923 C223.146464,209.540923 198.363093,223.878883 183.839389,245.643293 L183.952782,245.655104 C184.933157,244.762229 185.930063,243.885889 186.955321,243.03317 C222.033803,213.960416 272.668324,220.342816 300.338486,257.198698 Z M214.370819,264.53208 C220.980666,259.892912 228.629944,257.226098 236.796575,257.226098 C250.228874,257.226098 262.283922,264.413975 270.556862,275.811119 L288.30989,264.716324 L288.319343,264.716324 C280.157438,253.040453 266.61174,245.39669 251.277753,245.39669 C236.026448,245.39669 222.549266,252.95778 214.370819,264.53208 Z M166.789394,213.901363 C218.255462,173.164548 291.088955,184.079823 329.878678,238.171964 L330.136173,238.568797 L349.186133,226.701596 C328.31953,194.777784 292.263042,173.684211 251.278701,173.684211 C210.866048,173.684211 174.47174,194.572281 153.416152,226.633095 C157.283311,222.56083 162.29621,217.458689 166.789394,213.901363 Z" id="icn-audible" fill="#FFFFFF" mask="url(#mask-2)"></path>
|
||||||
|
</g>
|
||||||
|
</g>
|
||||||
|
</g>
|
||||||
|
</g>
|
||||||
|
</svg>
|
||||||
|
After Width: | Height: | Size: 2.3 KiB |
@@ -0,0 +1,154 @@
|
|||||||
|
/**
|
||||||
|
* Component: Adjust Search Terms Modal
|
||||||
|
* Documentation: documentation/admin-dashboard.md
|
||||||
|
*/
|
||||||
|
|
||||||
|
'use client';
|
||||||
|
|
||||||
|
import { useState } from 'react';
|
||||||
|
import { Modal } from '@/components/ui/Modal';
|
||||||
|
import { fetchWithAuth } from '@/lib/utils/api';
|
||||||
|
import { useToast } from '@/components/ui/Toast';
|
||||||
|
|
||||||
|
interface AdjustSearchTermsModalProps {
|
||||||
|
isOpen: boolean;
|
||||||
|
onClose: () => void;
|
||||||
|
requestId: string;
|
||||||
|
title: string;
|
||||||
|
author: string;
|
||||||
|
currentSearchTerms?: string | null;
|
||||||
|
onSuccess?: () => void;
|
||||||
|
}
|
||||||
|
|
||||||
|
export function AdjustSearchTermsModal({
|
||||||
|
isOpen,
|
||||||
|
onClose,
|
||||||
|
requestId,
|
||||||
|
title,
|
||||||
|
author,
|
||||||
|
currentSearchTerms,
|
||||||
|
onSuccess,
|
||||||
|
}: AdjustSearchTermsModalProps) {
|
||||||
|
const toast = useToast();
|
||||||
|
const [searchTerms, setSearchTerms] = useState(currentSearchTerms || title);
|
||||||
|
const [isSaving, setIsSaving] = useState(false);
|
||||||
|
const [isSavingAndSearching, setIsSavingAndSearching] = useState(false);
|
||||||
|
|
||||||
|
// Reset state when modal opens
|
||||||
|
const handleClose = () => {
|
||||||
|
setSearchTerms(currentSearchTerms || title);
|
||||||
|
onClose();
|
||||||
|
};
|
||||||
|
|
||||||
|
const save = async (triggerSearch: boolean) => {
|
||||||
|
const setter = triggerSearch ? setIsSavingAndSearching : setIsSaving;
|
||||||
|
setter(true);
|
||||||
|
|
||||||
|
try {
|
||||||
|
// If terms match the original title, clear the override
|
||||||
|
const termsToSave = searchTerms.trim() === title ? null : searchTerms.trim() || null;
|
||||||
|
|
||||||
|
const response = await fetchWithAuth(`/api/admin/requests/${requestId}/search-terms`, {
|
||||||
|
method: 'PATCH',
|
||||||
|
headers: { 'Content-Type': 'application/json' },
|
||||||
|
body: JSON.stringify({ searchTerms: termsToSave, triggerSearch }),
|
||||||
|
});
|
||||||
|
|
||||||
|
if (!response.ok) {
|
||||||
|
const errorData = await response.json();
|
||||||
|
throw new Error(errorData.message || 'Failed to update search terms');
|
||||||
|
}
|
||||||
|
|
||||||
|
const data = await response.json();
|
||||||
|
|
||||||
|
if (data.searchTriggered) {
|
||||||
|
toast.success('Search terms saved and search triggered');
|
||||||
|
} else {
|
||||||
|
toast.success('Search terms saved');
|
||||||
|
}
|
||||||
|
|
||||||
|
onSuccess?.();
|
||||||
|
onClose();
|
||||||
|
} catch (error) {
|
||||||
|
toast.error(`Failed to save: ${error instanceof Error ? error.message : 'Unknown error'}`);
|
||||||
|
} finally {
|
||||||
|
setter(false);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
const handleReset = () => {
|
||||||
|
setSearchTerms(title);
|
||||||
|
};
|
||||||
|
|
||||||
|
const isLoading = isSaving || isSavingAndSearching;
|
||||||
|
const hasChanges = searchTerms.trim() !== (currentSearchTerms || title);
|
||||||
|
const isCustom = searchTerms.trim() !== title;
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Modal isOpen={isOpen} onClose={handleClose} title="Adjust Search Terms" size="sm">
|
||||||
|
<div className="space-y-4">
|
||||||
|
{/* Original info */}
|
||||||
|
<div className="bg-gray-50 dark:bg-gray-900/50 rounded-lg p-3 space-y-1">
|
||||||
|
<div className="text-xs font-medium text-gray-500 dark:text-gray-400 uppercase tracking-wider">
|
||||||
|
Original Title
|
||||||
|
</div>
|
||||||
|
<div className="text-sm text-gray-900 dark:text-gray-100 font-medium">{title}</div>
|
||||||
|
<div className="text-xs text-gray-500 dark:text-gray-400">by {author}</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
{/* Search terms input */}
|
||||||
|
<div>
|
||||||
|
<label
|
||||||
|
htmlFor="search-terms"
|
||||||
|
className="block text-sm font-medium text-gray-700 dark:text-gray-300 mb-1.5"
|
||||||
|
>
|
||||||
|
Search Terms
|
||||||
|
</label>
|
||||||
|
<input
|
||||||
|
id="search-terms"
|
||||||
|
type="text"
|
||||||
|
value={searchTerms}
|
||||||
|
onChange={(e) => setSearchTerms(e.target.value)}
|
||||||
|
disabled={isLoading}
|
||||||
|
className="w-full px-3 py-2 border border-gray-300 dark:border-gray-600 rounded-lg bg-white dark:bg-gray-800 text-gray-900 dark:text-gray-100 text-sm focus:ring-2 focus:ring-blue-500 focus:border-transparent disabled:opacity-50"
|
||||||
|
placeholder="Enter custom search terms..."
|
||||||
|
/>
|
||||||
|
{isCustom && (
|
||||||
|
<button
|
||||||
|
onClick={handleReset}
|
||||||
|
disabled={isLoading}
|
||||||
|
className="mt-1.5 text-xs text-blue-600 dark:text-blue-400 hover:text-blue-700 dark:hover:text-blue-300 transition-colors disabled:opacity-50"
|
||||||
|
>
|
||||||
|
Reset to original title
|
||||||
|
</button>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
|
||||||
|
{/* Actions */}
|
||||||
|
<div className="flex items-center justify-end gap-2 pt-2 border-t border-gray-200 dark:border-gray-700">
|
||||||
|
<button
|
||||||
|
onClick={handleClose}
|
||||||
|
disabled={isLoading}
|
||||||
|
className="px-4 py-2 text-sm font-medium text-gray-700 dark:text-gray-300 bg-white dark:bg-gray-800 border border-gray-300 dark:border-gray-600 rounded-lg hover:bg-gray-50 dark:hover:bg-gray-700 transition-colors disabled:opacity-50"
|
||||||
|
>
|
||||||
|
Cancel
|
||||||
|
</button>
|
||||||
|
<button
|
||||||
|
onClick={() => save(false)}
|
||||||
|
disabled={isLoading || !searchTerms.trim()}
|
||||||
|
className="px-4 py-2 text-sm font-medium text-gray-700 dark:text-gray-300 bg-white dark:bg-gray-800 border border-gray-300 dark:border-gray-600 rounded-lg hover:bg-gray-50 dark:hover:bg-gray-700 transition-colors disabled:opacity-50"
|
||||||
|
>
|
||||||
|
{isSaving ? 'Saving...' : 'Save'}
|
||||||
|
</button>
|
||||||
|
<button
|
||||||
|
onClick={() => save(true)}
|
||||||
|
disabled={isLoading || !searchTerms.trim()}
|
||||||
|
className="px-4 py-2 text-sm font-medium text-white bg-blue-600 hover:bg-blue-700 rounded-lg transition-colors disabled:opacity-50"
|
||||||
|
>
|
||||||
|
{isSavingAndSearching ? 'Saving...' : 'Save & Search'}
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</Modal>
|
||||||
|
);
|
||||||
|
}
|
||||||
@@ -2,12 +2,13 @@
|
|||||||
* Component: Confirm Dialog
|
* Component: Confirm Dialog
|
||||||
* Documentation: documentation/frontend/components.md
|
* Documentation: documentation/frontend/components.md
|
||||||
*
|
*
|
||||||
* Reusable confirmation dialog for destructive actions
|
* Reusable confirmation dialog for destructive actions.
|
||||||
|
* Features: backdrop blur, smooth enter animation, Escape to close, focus trap, ARIA.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
'use client';
|
'use client';
|
||||||
|
|
||||||
import { Fragment } from 'react';
|
import React, { useEffect, useRef } from 'react';
|
||||||
|
|
||||||
export interface ConfirmDialogProps {
|
export interface ConfirmDialogProps {
|
||||||
isOpen: boolean;
|
isOpen: boolean;
|
||||||
@@ -30,99 +31,177 @@ export function ConfirmDialog({
|
|||||||
onConfirm,
|
onConfirm,
|
||||||
onCancel,
|
onCancel,
|
||||||
}: ConfirmDialogProps) {
|
}: ConfirmDialogProps) {
|
||||||
|
const cancelRef = useRef<HTMLButtonElement>(null);
|
||||||
|
const confirmRef = useRef<HTMLButtonElement>(null);
|
||||||
|
const dialogRef = useRef<HTMLDivElement>(null);
|
||||||
|
|
||||||
|
// Focus the cancel button on open (safer default for destructive dialogs)
|
||||||
|
useEffect(() => {
|
||||||
|
if (isOpen) {
|
||||||
|
// Small delay to let animation start before stealing focus
|
||||||
|
const t = setTimeout(() => cancelRef.current?.focus(), 50);
|
||||||
|
return () => clearTimeout(t);
|
||||||
|
}
|
||||||
|
}, [isOpen]);
|
||||||
|
|
||||||
|
// Escape to close + focus trap
|
||||||
|
useEffect(() => {
|
||||||
|
if (!isOpen) return;
|
||||||
|
|
||||||
|
const handleKeyDown = (e: KeyboardEvent) => {
|
||||||
|
if (e.key === 'Escape') {
|
||||||
|
e.preventDefault();
|
||||||
|
onCancel();
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Focus trap: tab cycles only within dialog
|
||||||
|
if (e.key === 'Tab') {
|
||||||
|
const focusable = dialogRef.current?.querySelectorAll<HTMLElement>(
|
||||||
|
'button, [href], input, select, textarea, [tabindex]:not([tabindex="-1"])'
|
||||||
|
);
|
||||||
|
if (!focusable || focusable.length === 0) return;
|
||||||
|
|
||||||
|
const first = focusable[0];
|
||||||
|
const last = focusable[focusable.length - 1];
|
||||||
|
|
||||||
|
if (e.shiftKey) {
|
||||||
|
if (document.activeElement === first) {
|
||||||
|
e.preventDefault();
|
||||||
|
last.focus();
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
if (document.activeElement === last) {
|
||||||
|
e.preventDefault();
|
||||||
|
first.focus();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
document.addEventListener('keydown', handleKeyDown);
|
||||||
|
return () => document.removeEventListener('keydown', handleKeyDown);
|
||||||
|
}, [isOpen, onCancel]);
|
||||||
|
|
||||||
|
// Prevent body scroll while open
|
||||||
|
useEffect(() => {
|
||||||
|
if (isOpen) {
|
||||||
|
document.body.style.overflow = 'hidden';
|
||||||
|
return () => { document.body.style.overflow = ''; };
|
||||||
|
}
|
||||||
|
}, [isOpen]);
|
||||||
|
|
||||||
if (!isOpen) return null;
|
if (!isOpen) return null;
|
||||||
|
|
||||||
const confirmButtonClasses =
|
const isDestructive = confirmVariant === 'danger';
|
||||||
confirmVariant === 'danger'
|
|
||||||
? 'bg-red-600 hover:bg-red-700 text-white'
|
|
||||||
: 'bg-blue-600 hover:bg-blue-700 text-white';
|
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div className="fixed inset-0 z-50 overflow-y-auto">
|
<div
|
||||||
|
role="dialog"
|
||||||
|
aria-modal="true"
|
||||||
|
aria-labelledby="confirm-dialog-title"
|
||||||
|
aria-describedby="confirm-dialog-desc"
|
||||||
|
className="fixed inset-0 z-50 flex items-center justify-center p-4"
|
||||||
|
>
|
||||||
{/* Backdrop */}
|
{/* Backdrop */}
|
||||||
<div
|
<div
|
||||||
className="fixed inset-0 bg-black bg-opacity-50 transition-opacity"
|
className="animate-dialog-backdrop fixed inset-0 bg-black/40 backdrop-blur-sm"
|
||||||
onClick={onCancel}
|
onClick={onCancel}
|
||||||
aria-hidden="true"
|
aria-hidden="true"
|
||||||
/>
|
/>
|
||||||
|
|
||||||
{/* Dialog */}
|
{/* Panel */}
|
||||||
<div className="flex min-h-full items-center justify-center p-4 text-center sm:p-0">
|
<div
|
||||||
<div className="relative transform overflow-hidden rounded-lg bg-white dark:bg-gray-800 text-left shadow-xl transition-all sm:my-8 sm:w-full sm:max-w-lg">
|
ref={dialogRef}
|
||||||
<div className="bg-white dark:bg-gray-800 px-4 pb-4 pt-5 sm:p-6 sm:pb-4">
|
className="animate-dialog-panel relative w-full max-w-sm rounded-2xl overflow-hidden bg-white dark:bg-gray-900 shadow-2xl ring-1 ring-black/10 dark:ring-white/10"
|
||||||
<div className="sm:flex sm:items-start">
|
>
|
||||||
{/* Icon */}
|
{/* Header */}
|
||||||
<div
|
<div className="px-6 pt-6 pb-4">
|
||||||
className={`mx-auto flex h-12 w-12 flex-shrink-0 items-center justify-center rounded-full ${
|
<div className="flex items-start gap-4">
|
||||||
confirmVariant === 'danger'
|
{/* Icon well */}
|
||||||
? 'bg-red-100 dark:bg-red-900'
|
<div className={`flex-shrink-0 flex items-center justify-center w-10 h-10 rounded-full ${
|
||||||
: 'bg-blue-100 dark:bg-blue-900'
|
isDestructive
|
||||||
} sm:mx-0 sm:h-10 sm:w-10`}
|
? 'bg-red-50 dark:bg-red-500/10'
|
||||||
>
|
: 'bg-blue-50 dark:bg-blue-500/10'
|
||||||
|
}`}>
|
||||||
|
{isDestructive ? (
|
||||||
<svg
|
<svg
|
||||||
className={`h-6 w-6 ${
|
className="w-5 h-5 text-red-500 dark:text-red-400"
|
||||||
confirmVariant === 'danger'
|
|
||||||
? 'text-red-600 dark:text-red-400'
|
|
||||||
: 'text-blue-600 dark:text-blue-400'
|
|
||||||
}`}
|
|
||||||
fill="none"
|
fill="none"
|
||||||
viewBox="0 0 24 24"
|
viewBox="0 0 24 24"
|
||||||
strokeWidth="1.5"
|
strokeWidth="1.75"
|
||||||
stroke="currentColor"
|
stroke="currentColor"
|
||||||
|
aria-hidden="true"
|
||||||
>
|
>
|
||||||
{confirmVariant === 'danger' ? (
|
<path
|
||||||
<path
|
strokeLinecap="round"
|
||||||
strokeLinecap="round"
|
strokeLinejoin="round"
|
||||||
strokeLinejoin="round"
|
d="M12 9v3.75m-9.303 3.376c-.866 1.5.217 3.374 1.948 3.374h14.71c1.73 0 2.813-1.874 1.948-3.374L13.949 3.378c-.866-1.5-3.032-1.5-3.898 0L2.697 16.126zM12 15.75h.007v.008H12v-.008z"
|
||||||
d="M12 9v3.75m-9.303 3.376c-.866 1.5.217 3.374 1.948 3.374h14.71c1.73 0 2.813-1.874 1.948-3.374L13.949 3.378c-.866-1.5-3.032-1.5-3.898 0L2.697 16.126zM12 15.75h.007v.008H12v-.008z"
|
/>
|
||||||
/>
|
|
||||||
) : (
|
|
||||||
<path
|
|
||||||
strokeLinecap="round"
|
|
||||||
strokeLinejoin="round"
|
|
||||||
d="M9.879 7.519c1.171-1.025 3.071-1.025 4.242 0 1.172 1.025 1.172 2.687 0 3.712-.203.179-.43.326-.67.442-.745.361-1.45.999-1.45 1.827v.75M21 12a9 9 0 11-18 0 9 9 0 0118 0zm-9 5.25h.008v.008H12v-.008z"
|
|
||||||
/>
|
|
||||||
)}
|
|
||||||
</svg>
|
</svg>
|
||||||
</div>
|
) : (
|
||||||
|
<svg
|
||||||
|
className="w-5 h-5 text-blue-500 dark:text-blue-400"
|
||||||
|
fill="none"
|
||||||
|
viewBox="0 0 24 24"
|
||||||
|
strokeWidth="1.75"
|
||||||
|
stroke="currentColor"
|
||||||
|
aria-hidden="true"
|
||||||
|
>
|
||||||
|
<path
|
||||||
|
strokeLinecap="round"
|
||||||
|
strokeLinejoin="round"
|
||||||
|
d="M9.879 7.519c1.171-1.025 3.071-1.025 4.242 0 1.172 1.025 1.172 2.687 0 3.712-.203.179-.43.326-.67.442-.745.361-1.45.999-1.45 1.827v.75M21 12a9 9 0 11-18 0 9 9 0 0118 0zm-9 5.25h.008v.008H12v-.008z"
|
||||||
|
/>
|
||||||
|
</svg>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
|
||||||
{/* Content */}
|
{/* Text */}
|
||||||
<div className="mt-3 text-center sm:ml-4 sm:mt-0 sm:text-left flex-1">
|
<div className="flex-1 min-w-0 pt-0.5">
|
||||||
<h3 className="text-lg font-semibold leading-6 text-gray-900 dark:text-gray-100">
|
<h3
|
||||||
{title}
|
id="confirm-dialog-title"
|
||||||
</h3>
|
className="text-base font-semibold leading-6 text-gray-900 dark:text-gray-50"
|
||||||
<div className="mt-2">
|
>
|
||||||
{typeof message === 'string' ? (
|
{title}
|
||||||
<p className="text-sm text-gray-500 dark:text-gray-400 whitespace-pre-line">
|
</h3>
|
||||||
{message}
|
<div id="confirm-dialog-desc" className="mt-1.5">
|
||||||
</p>
|
{typeof message === 'string' ? (
|
||||||
) : (
|
<p className="text-sm text-gray-500 dark:text-gray-400 leading-relaxed">
|
||||||
<div className="text-sm text-gray-500 dark:text-gray-400">
|
{message}
|
||||||
{message}
|
</p>
|
||||||
</div>
|
) : (
|
||||||
)}
|
<div className="text-sm text-gray-500 dark:text-gray-400 leading-relaxed">
|
||||||
</div>
|
{message}
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
{/* Actions */}
|
{/* Action bar */}
|
||||||
<div className="bg-gray-50 dark:bg-gray-900 px-4 py-3 sm:flex sm:flex-row-reverse sm:px-6 gap-2">
|
<div className="flex items-center justify-end gap-2 px-6 py-4 bg-gray-50/80 dark:bg-white/[0.03] border-t border-gray-100 dark:border-white/[0.06]">
|
||||||
<button
|
<button
|
||||||
type="button"
|
ref={cancelRef}
|
||||||
onClick={onConfirm}
|
type="button"
|
||||||
className={`inline-flex w-full justify-center rounded-lg px-4 py-2 text-sm font-semibold shadow-sm sm:w-auto transition-colors ${confirmButtonClasses}`}
|
onClick={onCancel}
|
||||||
>
|
className="px-4 py-2 text-sm font-medium rounded-xl text-gray-700 dark:text-gray-300 bg-white dark:bg-white/[0.06] hover:bg-gray-100 dark:hover:bg-white/[0.1] border border-gray-200 dark:border-white/[0.1] transition-all duration-150 focus:outline-none focus-visible:ring-2 focus-visible:ring-gray-400 focus-visible:ring-offset-2 dark:focus-visible:ring-offset-gray-900"
|
||||||
{confirmLabel}
|
>
|
||||||
</button>
|
{cancelLabel}
|
||||||
<button
|
</button>
|
||||||
type="button"
|
<button
|
||||||
onClick={onCancel}
|
ref={confirmRef}
|
||||||
className="mt-3 inline-flex w-full justify-center rounded-lg bg-white dark:bg-gray-700 px-4 py-2 text-sm font-semibold text-gray-900 dark:text-gray-100 shadow-sm ring-1 ring-inset ring-gray-300 dark:ring-gray-600 hover:bg-gray-50 dark:hover:bg-gray-600 sm:mt-0 sm:w-auto transition-colors"
|
type="button"
|
||||||
>
|
onClick={onConfirm}
|
||||||
{cancelLabel}
|
className={`px-4 py-2 text-sm font-medium rounded-xl text-white transition-all duration-150 focus:outline-none focus-visible:ring-2 focus-visible:ring-offset-2 dark:focus-visible:ring-offset-gray-900 active:scale-[0.97] ${
|
||||||
</button>
|
isDestructive
|
||||||
</div>
|
? 'bg-red-600 hover:bg-red-700 focus-visible:ring-red-500'
|
||||||
|
: 'bg-blue-600 hover:bg-blue-700 focus-visible:ring-blue-500'
|
||||||
|
}`}
|
||||||
|
>
|
||||||
|
{confirmLabel}
|
||||||
|
</button>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|||||||
@@ -28,6 +28,8 @@ interface RecentRequest {
|
|||||||
completedAt: Date | null;
|
completedAt: Date | null;
|
||||||
errorMessage: string | null;
|
errorMessage: string | null;
|
||||||
torrentUrl?: string | null;
|
torrentUrl?: string | null;
|
||||||
|
downloadAttempts?: number;
|
||||||
|
customSearchTerms?: string | null;
|
||||||
}
|
}
|
||||||
|
|
||||||
interface User {
|
interface User {
|
||||||
@@ -161,7 +163,7 @@ function getInitialParams(): {
|
|||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
export function RecentRequestsTable({ ebookSidecarEnabled = false, annasArchiveBaseUrl = 'https://annas-archive.li' }: RecentRequestsTableProps) {
|
export function RecentRequestsTable({ ebookSidecarEnabled = false, annasArchiveBaseUrl = 'https://annas-archive.gl' }: RecentRequestsTableProps) {
|
||||||
const toast = useToast();
|
const toast = useToast();
|
||||||
|
|
||||||
// Get initial filter state from URL (only evaluated once due to lazy init)
|
// Get initial filter state from URL (only evaluated once due to lazy init)
|
||||||
@@ -444,6 +446,29 @@ export function RecentRequestsTable({ ebookSidecarEnabled = false, annasArchiveB
|
|||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
const handleRetryDownload = async (requestId: string) => {
|
||||||
|
try {
|
||||||
|
const response = await fetchWithAuth(`/api/admin/requests/${requestId}/retry-download`, {
|
||||||
|
method: 'POST',
|
||||||
|
headers: {
|
||||||
|
'Content-Type': 'application/json',
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
const responseData = await response.json();
|
||||||
|
|
||||||
|
if (!response.ok) {
|
||||||
|
throw new Error(responseData.message || 'Failed to retry download');
|
||||||
|
}
|
||||||
|
|
||||||
|
toast.success(responseData.message || 'Download retry initiated');
|
||||||
|
await mutate(apiUrl);
|
||||||
|
} catch (error) {
|
||||||
|
console.error('[Admin] Failed to retry download:', error);
|
||||||
|
toast.error(`Failed to retry download: ${error instanceof Error ? error.message : 'Unknown error'}`);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
// Render loading state
|
// Render loading state
|
||||||
if (isLoading && !data) {
|
if (isLoading && !data) {
|
||||||
return (
|
return (
|
||||||
@@ -638,6 +663,17 @@ export function RecentRequestsTable({ ebookSidecarEnabled = false, annasArchiveB
|
|||||||
Ebook
|
Ebook
|
||||||
</span>
|
</span>
|
||||||
)}
|
)}
|
||||||
|
{request.customSearchTerms && (
|
||||||
|
<span
|
||||||
|
className="inline-flex items-center gap-1 px-2 py-0.5 text-xs font-medium rounded-full bg-blue-100 text-blue-700 dark:bg-blue-900 dark:text-blue-200"
|
||||||
|
title={`Custom search: ${request.customSearchTerms}`}
|
||||||
|
>
|
||||||
|
<svg className="w-3 h-3" fill="none" stroke="currentColor" viewBox="0 0 24 24">
|
||||||
|
<path strokeLinecap="round" strokeLinejoin="round" strokeWidth={2} d="M11 5H6a2 2 0 00-2 2v11a2 2 0 002 2h11a2 2 0 002-2v-5m-1.414-9.414a2 2 0 112.828 2.828L11.828 15H9v-2.828l8.586-8.586z" />
|
||||||
|
</svg>
|
||||||
|
Custom Search
|
||||||
|
</span>
|
||||||
|
)}
|
||||||
</div>
|
</div>
|
||||||
<div className="text-sm text-gray-500 dark:text-gray-400">
|
<div className="text-sm text-gray-500 dark:text-gray-400">
|
||||||
{request.author}
|
{request.author}
|
||||||
@@ -673,12 +709,16 @@ export function RecentRequestsTable({ ebookSidecarEnabled = false, annasArchiveB
|
|||||||
type: request.type,
|
type: request.type,
|
||||||
asin: request.asin,
|
asin: request.asin,
|
||||||
torrentUrl: request.torrentUrl,
|
torrentUrl: request.torrentUrl,
|
||||||
|
downloadAttempts: request.downloadAttempts,
|
||||||
|
customSearchTerms: request.customSearchTerms,
|
||||||
}}
|
}}
|
||||||
onDelete={handleDeleteClick}
|
onDelete={handleDeleteClick}
|
||||||
onManualSearch={handleManualSearch}
|
onManualSearch={handleManualSearch}
|
||||||
onCancel={handleCancel}
|
onCancel={handleCancel}
|
||||||
|
onRetryDownload={handleRetryDownload}
|
||||||
onViewDetails={(asin) => handleViewDetails(asin, request.status)}
|
onViewDetails={(asin) => handleViewDetails(asin, request.status)}
|
||||||
onFetchEbook={handleFetchEbook}
|
onFetchEbook={handleFetchEbook}
|
||||||
|
onSearchTermsUpdated={() => mutate(apiUrl)}
|
||||||
ebookSidecarEnabled={ebookSidecarEnabled}
|
ebookSidecarEnabled={ebookSidecarEnabled}
|
||||||
annasArchiveBaseUrl={annasArchiveBaseUrl}
|
annasArchiveBaseUrl={annasArchiveBaseUrl}
|
||||||
isLoading={isDeleting || isFetchingEbook}
|
isLoading={isDeleting || isFetchingEbook}
|
||||||
@@ -835,7 +875,6 @@ export function RecentRequestsTable({ ebookSidecarEnabled = false, annasArchiveB
|
|||||||
}}
|
}}
|
||||||
isAvailable={viewDetailsStatus === 'available' || viewDetailsStatus === 'completed'}
|
isAvailable={viewDetailsStatus === 'available' || viewDetailsStatus === 'completed'}
|
||||||
requestStatus={viewDetailsStatus}
|
requestStatus={viewDetailsStatus}
|
||||||
hideRequestActions
|
|
||||||
/>
|
/>
|
||||||
)}
|
)}
|
||||||
</div>
|
</div>
|
||||||
|
|||||||
@@ -114,23 +114,13 @@ export function ReportedIssuesSection({ issues }: ReportedIssuesSectionProps) {
|
|||||||
<div className="flex gap-3">
|
<div className="flex gap-3">
|
||||||
{/* Cover Image */}
|
{/* Cover Image */}
|
||||||
<div className="flex-shrink-0">
|
<div className="flex-shrink-0">
|
||||||
{issue.audiobook.coverArtUrl ? (
|
{/* eslint-disable-next-line @next/next/no-img-element */}
|
||||||
<img
|
<img
|
||||||
src={issue.audiobook.coverArtUrl}
|
src={issue.audiobook.coverArtUrl || '/placeholder_cover.svg'}
|
||||||
alt={issue.audiobook.title}
|
alt={issue.audiobook.title}
|
||||||
className="w-16 h-16 rounded object-cover"
|
className="w-16 h-16 rounded object-cover"
|
||||||
/>
|
onError={(e) => { (e.target as HTMLImageElement).src = '/placeholder_cover.svg'; }}
|
||||||
) : (
|
/>
|
||||||
<div className="w-16 h-16 rounded bg-gray-200 dark:bg-gray-700 flex items-center justify-center">
|
|
||||||
<svg
|
|
||||||
className="w-8 h-8 text-gray-400 dark:text-gray-600"
|
|
||||||
fill="currentColor"
|
|
||||||
viewBox="0 0 20 20"
|
|
||||||
>
|
|
||||||
<path d="M9 4.804A7.968 7.968 0 005.5 4c-1.255 0-2.443.29-3.5.804v10A7.969 7.969 0 015.5 14c1.669 0 3.218.51 4.5 1.385A7.962 7.962 0 0114.5 14c1.255 0 2.443.29 3.5.804v-10A7.968 7.968 0 0014.5 4c-1.255 0-2.443.29-3.5.804V12a1 1 0 11-2 0V4.804z" />
|
|
||||||
</svg>
|
|
||||||
</div>
|
|
||||||
)}
|
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
{/* Info */}
|
{/* Info */}
|
||||||
|
|||||||
@@ -10,6 +10,7 @@
|
|||||||
import { useState, useRef, useEffect } from 'react';
|
import { useState, useRef, useEffect } from 'react';
|
||||||
import { createPortal } from 'react-dom';
|
import { createPortal } from 'react-dom';
|
||||||
import { InteractiveTorrentSearchModal } from '@/components/requests/InteractiveTorrentSearchModal';
|
import { InteractiveTorrentSearchModal } from '@/components/requests/InteractiveTorrentSearchModal';
|
||||||
|
import { AdjustSearchTermsModal } from './AdjustSearchTermsModal';
|
||||||
import { useSmartDropdownPosition } from '@/hooks/useSmartDropdownPosition';
|
import { useSmartDropdownPosition } from '@/hooks/useSmartDropdownPosition';
|
||||||
|
|
||||||
export interface RequestActionsDropdownProps {
|
export interface RequestActionsDropdownProps {
|
||||||
@@ -21,12 +22,16 @@ export interface RequestActionsDropdownProps {
|
|||||||
type?: 'audiobook' | 'ebook';
|
type?: 'audiobook' | 'ebook';
|
||||||
asin?: string | null;
|
asin?: string | null;
|
||||||
torrentUrl?: string | null;
|
torrentUrl?: string | null;
|
||||||
|
downloadAttempts?: number;
|
||||||
|
customSearchTerms?: string | null;
|
||||||
};
|
};
|
||||||
onDelete: (requestId: string, title: string) => void;
|
onDelete: (requestId: string, title: string) => void;
|
||||||
onManualSearch: (requestId: string) => Promise<void>;
|
onManualSearch: (requestId: string) => Promise<void>;
|
||||||
onCancel: (requestId: string) => Promise<void>;
|
onCancel: (requestId: string) => Promise<void>;
|
||||||
|
onRetryDownload?: (requestId: string) => Promise<void>;
|
||||||
onViewDetails?: (asin: string) => void;
|
onViewDetails?: (asin: string) => void;
|
||||||
onFetchEbook?: (requestId: string) => Promise<void>;
|
onFetchEbook?: (requestId: string) => Promise<void>;
|
||||||
|
onSearchTermsUpdated?: () => void;
|
||||||
ebookSidecarEnabled?: boolean;
|
ebookSidecarEnabled?: boolean;
|
||||||
annasArchiveBaseUrl?: string;
|
annasArchiveBaseUrl?: string;
|
||||||
isLoading?: boolean;
|
isLoading?: boolean;
|
||||||
@@ -37,15 +42,18 @@ export function RequestActionsDropdown({
|
|||||||
onDelete,
|
onDelete,
|
||||||
onManualSearch,
|
onManualSearch,
|
||||||
onCancel,
|
onCancel,
|
||||||
|
onRetryDownload,
|
||||||
onViewDetails,
|
onViewDetails,
|
||||||
onFetchEbook,
|
onFetchEbook,
|
||||||
|
onSearchTermsUpdated,
|
||||||
ebookSidecarEnabled = false,
|
ebookSidecarEnabled = false,
|
||||||
annasArchiveBaseUrl = 'https://annas-archive.li',
|
annasArchiveBaseUrl = 'https://annas-archive.gl',
|
||||||
isLoading = false,
|
isLoading = false,
|
||||||
}: RequestActionsDropdownProps) {
|
}: RequestActionsDropdownProps) {
|
||||||
const [isOpen, setIsOpen] = useState(false);
|
const [isOpen, setIsOpen] = useState(false);
|
||||||
const [showInteractiveSearch, setShowInteractiveSearch] = useState(false);
|
const [showInteractiveSearch, setShowInteractiveSearch] = useState(false);
|
||||||
const [showInteractiveSearchEbook, setShowInteractiveSearchEbook] = useState(false);
|
const [showInteractiveSearchEbook, setShowInteractiveSearchEbook] = useState(false);
|
||||||
|
const [showAdjustSearchTerms, setShowAdjustSearchTerms] = useState(false);
|
||||||
const { containerRef, dropdownRef, positionAbove, style } = useSmartDropdownPosition(isOpen);
|
const { containerRef, dropdownRef, positionAbove, style } = useSmartDropdownPosition(isOpen);
|
||||||
|
|
||||||
// Determine request type
|
// Determine request type
|
||||||
@@ -54,9 +62,10 @@ export function RequestActionsDropdown({
|
|||||||
// View Details: available when ASIN exists (audiobook requests only)
|
// View Details: available when ASIN exists (audiobook requests only)
|
||||||
const canViewDetails = !isEbook && !!request.asin && !!onViewDetails;
|
const canViewDetails = !isEbook && !!request.asin && !!onViewDetails;
|
||||||
|
|
||||||
// Determine available actions based on status and type
|
// Determine available actions based on status
|
||||||
// Ebooks don't support manual/interactive search (Anna's Archive only)
|
const canSearch = ['pending', 'failed', 'awaiting_search'].includes(request.status);
|
||||||
const canSearch = !isEbook && ['pending', 'failed', 'awaiting_search'].includes(request.status);
|
const canAdjustSearchTerms = ['pending', 'failed', 'awaiting_search', 'searching'].includes(request.status);
|
||||||
|
const canRetryDownload = request.status === 'failed' && (request.downloadAttempts ?? 0) > 0 && !!onRetryDownload;
|
||||||
const canCancel = ['pending', 'searching', 'downloading'].includes(request.status);
|
const canCancel = ['pending', 'searching', 'downloading'].includes(request.status);
|
||||||
const canDelete = true; // Admins can always delete
|
const canDelete = true; // Admins can always delete
|
||||||
|
|
||||||
@@ -120,7 +129,16 @@ export function RequestActionsDropdown({
|
|||||||
|
|
||||||
const handleInteractiveSearch = () => {
|
const handleInteractiveSearch = () => {
|
||||||
setIsOpen(false);
|
setIsOpen(false);
|
||||||
setShowInteractiveSearch(true);
|
if (isEbook) {
|
||||||
|
setShowInteractiveSearchEbook(true);
|
||||||
|
} else {
|
||||||
|
setShowInteractiveSearch(true);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
const handleAdjustSearchTerms = () => {
|
||||||
|
setIsOpen(false);
|
||||||
|
setShowAdjustSearchTerms(true);
|
||||||
};
|
};
|
||||||
|
|
||||||
const handleInteractiveSearchEbook = () => {
|
const handleInteractiveSearchEbook = () => {
|
||||||
@@ -128,6 +146,17 @@ export function RequestActionsDropdown({
|
|||||||
setShowInteractiveSearchEbook(true);
|
setShowInteractiveSearchEbook(true);
|
||||||
};
|
};
|
||||||
|
|
||||||
|
const handleRetryDownload = async () => {
|
||||||
|
setIsOpen(false);
|
||||||
|
if (onRetryDownload) {
|
||||||
|
try {
|
||||||
|
await onRetryDownload(request.requestId);
|
||||||
|
} catch (error) {
|
||||||
|
console.error('Failed to retry download:', error);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
const handleCancel = async () => {
|
const handleCancel = async () => {
|
||||||
setIsOpen(false);
|
setIsOpen(false);
|
||||||
if (window.confirm(`Are you sure you want to cancel the request for "${request.title}"?`)) {
|
if (window.confirm(`Are you sure you want to cancel the request for "${request.title}"?`)) {
|
||||||
@@ -253,6 +282,35 @@ export function RequestActionsDropdown({
|
|||||||
</button>
|
</button>
|
||||||
)}
|
)}
|
||||||
|
|
||||||
|
{/* Adjust Search Terms */}
|
||||||
|
{canAdjustSearchTerms && (
|
||||||
|
<button
|
||||||
|
onClick={handleAdjustSearchTerms}
|
||||||
|
className="w-full text-left px-4 py-2 text-sm text-gray-700 dark:text-gray-200 hover:bg-gray-100 dark:hover:bg-gray-700 flex items-center gap-2 transition-colors"
|
||||||
|
role="menuitem"
|
||||||
|
>
|
||||||
|
<svg
|
||||||
|
className="w-4 h-4"
|
||||||
|
fill="none"
|
||||||
|
stroke="currentColor"
|
||||||
|
viewBox="0 0 24 24"
|
||||||
|
>
|
||||||
|
<path
|
||||||
|
strokeLinecap="round"
|
||||||
|
strokeLinejoin="round"
|
||||||
|
strokeWidth={2}
|
||||||
|
d="M11 5H6a2 2 0 00-2 2v11a2 2 0 002 2h11a2 2 0 002-2v-5m-1.414-9.414a2 2 0 112.828 2.828L11.828 15H9v-2.828l8.586-8.586z"
|
||||||
|
/>
|
||||||
|
</svg>
|
||||||
|
<span className="flex items-center gap-1.5">
|
||||||
|
Adjust Search Terms
|
||||||
|
{request.customSearchTerms && (
|
||||||
|
<span className="w-1.5 h-1.5 rounded-full bg-blue-500 flex-shrink-0" />
|
||||||
|
)}
|
||||||
|
</span>
|
||||||
|
</button>
|
||||||
|
)}
|
||||||
|
|
||||||
{/* View Source */}
|
{/* View Source */}
|
||||||
{canViewSource && viewSourceUrl && (
|
{canViewSource && viewSourceUrl && (
|
||||||
<a
|
<a
|
||||||
@@ -328,8 +386,32 @@ export function RequestActionsDropdown({
|
|||||||
</button>
|
</button>
|
||||||
)}
|
)}
|
||||||
|
|
||||||
{/* Divider if we have search/view actions and other actions */}
|
{/* Retry Download */}
|
||||||
{(canSearch || canViewSource || canFetchEbook) && (canCancel || canDelete) && (
|
{canRetryDownload && (
|
||||||
|
<button
|
||||||
|
onClick={handleRetryDownload}
|
||||||
|
className="w-full text-left px-4 py-2 text-sm text-gray-700 dark:text-gray-200 hover:bg-gray-100 dark:hover:bg-gray-700 flex items-center gap-2 transition-colors"
|
||||||
|
role="menuitem"
|
||||||
|
>
|
||||||
|
<svg
|
||||||
|
className="w-4 h-4"
|
||||||
|
fill="none"
|
||||||
|
stroke="currentColor"
|
||||||
|
viewBox="0 0 24 24"
|
||||||
|
>
|
||||||
|
<path
|
||||||
|
strokeLinecap="round"
|
||||||
|
strokeLinejoin="round"
|
||||||
|
strokeWidth={2}
|
||||||
|
d="M4 16v1a3 3 0 003 3h10a3 3 0 003-3v-1m-4-4l-4 4m0 0l-4-4m4 4V4"
|
||||||
|
/>
|
||||||
|
</svg>
|
||||||
|
Retry Download
|
||||||
|
</button>
|
||||||
|
)}
|
||||||
|
|
||||||
|
{/* Divider if we have search/view/retry actions and other actions */}
|
||||||
|
{(canSearch || canViewSource || canFetchEbook || canRetryDownload) && (canCancel || canDelete) && (
|
||||||
<div className="border-t border-gray-200 dark:border-gray-700 my-1" />
|
<div className="border-t border-gray-200 dark:border-gray-700 my-1" />
|
||||||
)}
|
)}
|
||||||
|
|
||||||
@@ -358,7 +440,7 @@ export function RequestActionsDropdown({
|
|||||||
)}
|
)}
|
||||||
|
|
||||||
{/* Divider before delete */}
|
{/* Divider before delete */}
|
||||||
{canDelete && (canSearch || canCancel) && (
|
{canDelete && (canSearch || canRetryDownload || canCancel) && (
|
||||||
<div className="border-t border-gray-200 dark:border-gray-700 my-1" />
|
<div className="border-t border-gray-200 dark:border-gray-700 my-1" />
|
||||||
)}
|
)}
|
||||||
|
|
||||||
@@ -421,6 +503,7 @@ export function RequestActionsDropdown({
|
|||||||
title: request.title,
|
title: request.title,
|
||||||
author: request.author,
|
author: request.author,
|
||||||
}}
|
}}
|
||||||
|
customSearchTerms={request.customSearchTerms}
|
||||||
/>
|
/>
|
||||||
|
|
||||||
{/* Interactive Search Modal (Ebook) */}
|
{/* Interactive Search Modal (Ebook) */}
|
||||||
@@ -433,6 +516,18 @@ export function RequestActionsDropdown({
|
|||||||
author: request.author,
|
author: request.author,
|
||||||
}}
|
}}
|
||||||
searchMode="ebook"
|
searchMode="ebook"
|
||||||
|
customSearchTerms={request.customSearchTerms}
|
||||||
|
/>
|
||||||
|
|
||||||
|
{/* Adjust Search Terms Modal */}
|
||||||
|
<AdjustSearchTermsModal
|
||||||
|
isOpen={showAdjustSearchTerms}
|
||||||
|
onClose={() => setShowAdjustSearchTerms(false)}
|
||||||
|
requestId={request.requestId}
|
||||||
|
title={request.title}
|
||||||
|
author={request.author}
|
||||||
|
currentSearchTerms={request.customSearchTerms}
|
||||||
|
onSuccess={onSearchTermsUpdated}
|
||||||
/>
|
/>
|
||||||
</>
|
</>
|
||||||
);
|
);
|
||||||
|
|||||||
+164
-87
@@ -13,17 +13,34 @@ import { ActiveDownloadsTable } from './components/ActiveDownloadsTable';
|
|||||||
import { RecentRequestsTable } from './components/RecentRequestsTable';
|
import { RecentRequestsTable } from './components/RecentRequestsTable';
|
||||||
import { ToastProvider, useToast } from '@/components/ui/Toast';
|
import { ToastProvider, useToast } from '@/components/ui/Toast';
|
||||||
import { ReportedIssuesSection } from './components/ReportedIssuesSection';
|
import { ReportedIssuesSection } from './components/ReportedIssuesSection';
|
||||||
|
import { InteractiveTorrentSearchModal } from '@/components/requests/InteractiveTorrentSearchModal';
|
||||||
|
import { TorrentResult } from '@/lib/utils/ranking-algorithm';
|
||||||
import { formatDistanceToNow } from 'date-fns';
|
import { formatDistanceToNow } from 'date-fns';
|
||||||
import { useState } from 'react';
|
import { useState } from 'react';
|
||||||
|
|
||||||
|
interface SelectedTorrentData {
|
||||||
|
title?: string;
|
||||||
|
indexer?: string;
|
||||||
|
size?: number;
|
||||||
|
format?: string;
|
||||||
|
ebookFormat?: string;
|
||||||
|
seeders?: number;
|
||||||
|
infoUrl?: string;
|
||||||
|
source?: string;
|
||||||
|
protocol?: string;
|
||||||
|
score?: number;
|
||||||
|
}
|
||||||
|
|
||||||
interface PendingApprovalRequest {
|
interface PendingApprovalRequest {
|
||||||
id: string;
|
id: string;
|
||||||
createdAt: string;
|
createdAt: string;
|
||||||
type: 'audiobook' | 'ebook';
|
type: 'audiobook' | 'ebook';
|
||||||
|
selectedTorrent: SelectedTorrentData | null;
|
||||||
audiobook: {
|
audiobook: {
|
||||||
title: string;
|
title: string;
|
||||||
author: string;
|
author: string;
|
||||||
coverArtUrl: string | null;
|
coverArtUrl: string | null;
|
||||||
|
audibleAsin: string | null;
|
||||||
};
|
};
|
||||||
user: {
|
user: {
|
||||||
id: string;
|
id: string;
|
||||||
@@ -32,9 +49,20 @@ interface PendingApprovalRequest {
|
|||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
|
function formatTorrentSize(bytes: number): string {
|
||||||
|
const gb = bytes / (1024 ** 3);
|
||||||
|
const mb = bytes / (1024 ** 2);
|
||||||
|
return gb >= 1 ? `${gb.toFixed(1)} GB` : `${mb.toFixed(0)} MB`;
|
||||||
|
}
|
||||||
|
|
||||||
function PendingApprovalSection({ requests }: { requests: PendingApprovalRequest[] }) {
|
function PendingApprovalSection({ requests }: { requests: PendingApprovalRequest[] }) {
|
||||||
const toast = useToast();
|
const toast = useToast();
|
||||||
const [loadingStates, setLoadingStates] = useState<Record<string, boolean>>({});
|
const [loadingStates, setLoadingStates] = useState<Record<string, boolean>>({});
|
||||||
|
const [searchModalRequestId, setSearchModalRequestId] = useState<string | null>(null);
|
||||||
|
|
||||||
|
const searchModalRequest = searchModalRequestId
|
||||||
|
? requests.find((r) => r.id === searchModalRequestId)
|
||||||
|
: null;
|
||||||
|
|
||||||
const handleApproveRequest = async (requestId: string) => {
|
const handleApproveRequest = async (requestId: string) => {
|
||||||
setLoadingStates((prev) => ({ ...prev, [requestId]: true }));
|
setLoadingStates((prev) => ({ ...prev, [requestId]: true }));
|
||||||
@@ -47,7 +75,6 @@ function PendingApprovalSection({ requests }: { requests: PendingApprovalRequest
|
|||||||
|
|
||||||
toast.success('Request approved');
|
toast.success('Request approved');
|
||||||
|
|
||||||
// Mutate both pending requests and recent requests caches
|
|
||||||
await mutate('/api/admin/requests/pending-approval');
|
await mutate('/api/admin/requests/pending-approval');
|
||||||
await mutate('/api/admin/requests/recent');
|
await mutate('/api/admin/requests/recent');
|
||||||
await mutate('/api/admin/metrics');
|
await mutate('/api/admin/metrics');
|
||||||
@@ -72,7 +99,6 @@ function PendingApprovalSection({ requests }: { requests: PendingApprovalRequest
|
|||||||
|
|
||||||
toast.success('Request denied');
|
toast.success('Request denied');
|
||||||
|
|
||||||
// Mutate pending requests cache
|
|
||||||
await mutate('/api/admin/requests/pending-approval');
|
await mutate('/api/admin/requests/pending-approval');
|
||||||
await mutate('/api/admin/metrics');
|
await mutate('/api/admin/metrics');
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
@@ -85,6 +111,26 @@ function PendingApprovalSection({ requests }: { requests: PendingApprovalRequest
|
|||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
const handleApproveWithTorrent = async (requestId: string, torrent: TorrentResult) => {
|
||||||
|
await fetchJSON(`/api/admin/requests/${requestId}/approve`, {
|
||||||
|
method: 'POST',
|
||||||
|
body: JSON.stringify({ action: 'approve', selectedTorrent: torrent }),
|
||||||
|
});
|
||||||
|
|
||||||
|
toast.success('Request approved and download started');
|
||||||
|
|
||||||
|
await mutate('/api/admin/requests/pending-approval');
|
||||||
|
await mutate('/api/admin/requests/recent');
|
||||||
|
await mutate('/api/admin/metrics');
|
||||||
|
};
|
||||||
|
|
||||||
|
const LoadingSpinner = () => (
|
||||||
|
<svg className="animate-spin h-4 w-4" fill="none" viewBox="0 0 24 24">
|
||||||
|
<circle className="opacity-25" cx="12" cy="12" r="10" stroke="currentColor" strokeWidth="4" />
|
||||||
|
<path className="opacity-75" fill="currentColor" d="M4 12a8 8 0 018-8V0C5.373 0 0 5.373 0 12h4zm2 5.291A7.962 7.962 0 014 12H0c0 3.042 1.135 5.824 3 7.938l3-2.647z" />
|
||||||
|
</svg>
|
||||||
|
);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div className="mb-8">
|
<div className="mb-8">
|
||||||
{/* Section Header */}
|
{/* Section Header */}
|
||||||
@@ -116,6 +162,9 @@ function PendingApprovalSection({ requests }: { requests: PendingApprovalRequest
|
|||||||
<div className="grid grid-cols-1 md:grid-cols-2 lg:grid-cols-3 gap-4">
|
<div className="grid grid-cols-1 md:grid-cols-2 lg:grid-cols-3 gap-4">
|
||||||
{requests.map((request) => {
|
{requests.map((request) => {
|
||||||
const isLoading = loadingStates[request.id] || false;
|
const isLoading = loadingStates[request.id] || false;
|
||||||
|
const torrent = request.selectedTorrent;
|
||||||
|
const displayFormat = torrent?.format || torrent?.ebookFormat;
|
||||||
|
const isAnnasArchive = torrent?.source === 'annas_archive';
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div
|
<div
|
||||||
@@ -127,23 +176,13 @@ function PendingApprovalSection({ requests }: { requests: PendingApprovalRequest
|
|||||||
<div className="flex gap-3">
|
<div className="flex gap-3">
|
||||||
{/* Cover Image */}
|
{/* Cover Image */}
|
||||||
<div className="flex-shrink-0">
|
<div className="flex-shrink-0">
|
||||||
{request.audiobook.coverArtUrl ? (
|
{/* eslint-disable-next-line @next/next/no-img-element */}
|
||||||
<img
|
<img
|
||||||
src={request.audiobook.coverArtUrl}
|
src={request.audiobook.coverArtUrl || '/placeholder_cover.svg'}
|
||||||
alt={request.audiobook.title}
|
alt={request.audiobook.title}
|
||||||
className="w-16 h-16 rounded object-cover"
|
className="w-16 h-16 rounded object-cover"
|
||||||
/>
|
onError={(e) => { (e.target as HTMLImageElement).src = '/placeholder_cover.svg'; }}
|
||||||
) : (
|
/>
|
||||||
<div className="w-16 h-16 rounded bg-gray-200 dark:bg-gray-700 flex items-center justify-center">
|
|
||||||
<svg
|
|
||||||
className="w-8 h-8 text-gray-400 dark:text-gray-600"
|
|
||||||
fill="currentColor"
|
|
||||||
viewBox="0 0 20 20"
|
|
||||||
>
|
|
||||||
<path d="M9 4.804A7.968 7.968 0 005.5 4c-1.255 0-2.443.29-3.5.804v10A7.969 7.969 0 015.5 14c1.669 0 3.218.51 4.5 1.385A7.962 7.962 0 0114.5 14c1.255 0 2.443.29 3.5.804v-10A7.968 7.968 0 0014.5 4c-1.255 0-2.443.29-3.5.804V12a1 1 0 11-2 0V4.804z" />
|
|
||||||
</svg>
|
|
||||||
</div>
|
|
||||||
)}
|
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
{/* Book Info */}
|
{/* Book Info */}
|
||||||
@@ -205,89 +244,107 @@ function PendingApprovalSection({ requests }: { requests: PendingApprovalRequest
|
|||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
|
{/* Pre-Selected Release */}
|
||||||
|
{torrent && torrent.title && (
|
||||||
|
<div className="mx-4 mb-3 px-3 py-2.5 bg-gray-50 dark:bg-gray-900/60 rounded-lg border border-gray-200 dark:border-gray-700/60">
|
||||||
|
<div className="flex items-center gap-1.5 mb-1">
|
||||||
|
<svg className="w-3 h-3 text-gray-400 dark:text-gray-500 flex-shrink-0" fill="none" stroke="currentColor" viewBox="0 0 24 24">
|
||||||
|
<path strokeLinecap="round" strokeLinejoin="round" strokeWidth={2} d="M13.828 10.172a4 4 0 00-5.656 0l-4 4a4 4 0 105.656 5.656l1.102-1.101m-.758-4.899a4 4 0 005.656 0l4-4a4 4 0 00-5.656-5.656l-1.1 1.1" />
|
||||||
|
</svg>
|
||||||
|
<span className="text-[10px] font-semibold uppercase tracking-wider text-gray-400 dark:text-gray-500">
|
||||||
|
User-Selected Release
|
||||||
|
</span>
|
||||||
|
</div>
|
||||||
|
{torrent.infoUrl ? (
|
||||||
|
<a
|
||||||
|
href={torrent.infoUrl}
|
||||||
|
target="_blank"
|
||||||
|
rel="noopener noreferrer"
|
||||||
|
className="text-xs font-medium text-blue-600 dark:text-blue-400 hover:text-blue-700 dark:hover:text-blue-300 transition-colors line-clamp-2 leading-snug"
|
||||||
|
title={torrent.title}
|
||||||
|
>
|
||||||
|
{torrent.title}
|
||||||
|
</a>
|
||||||
|
) : (
|
||||||
|
<p className="text-xs font-medium text-gray-700 dark:text-gray-300 line-clamp-2 leading-snug" title={torrent.title}>
|
||||||
|
{torrent.title}
|
||||||
|
</p>
|
||||||
|
)}
|
||||||
|
<div className="flex items-center gap-1 mt-1.5 text-[11px] text-gray-500 dark:text-gray-400 flex-wrap">
|
||||||
|
{isAnnasArchive ? (
|
||||||
|
<span className="text-orange-600 dark:text-orange-400 font-medium">Anna's Archive</span>
|
||||||
|
) : torrent.indexer ? (
|
||||||
|
<span>{torrent.indexer}</span>
|
||||||
|
) : null}
|
||||||
|
{torrent.size && torrent.size > 0 ? (
|
||||||
|
<>
|
||||||
|
<span className="text-gray-300 dark:text-gray-600 select-none">·</span>
|
||||||
|
<span>{formatTorrentSize(torrent.size)}</span>
|
||||||
|
</>
|
||||||
|
) : null}
|
||||||
|
{displayFormat ? (
|
||||||
|
<>
|
||||||
|
<span className="text-gray-300 dark:text-gray-600 select-none">·</span>
|
||||||
|
<span className="px-1 py-px text-[10px] font-semibold uppercase tracking-wide rounded bg-purple-100 dark:bg-purple-500/15 text-purple-700 dark:text-purple-300">
|
||||||
|
{displayFormat}
|
||||||
|
</span>
|
||||||
|
</>
|
||||||
|
) : null}
|
||||||
|
{torrent.protocol === 'usenet' ? (
|
||||||
|
<>
|
||||||
|
<span className="text-gray-300 dark:text-gray-600 select-none">·</span>
|
||||||
|
<span className="text-sky-600 dark:text-sky-400 font-medium">NZB</span>
|
||||||
|
</>
|
||||||
|
) : torrent.seeders !== undefined && torrent.seeders !== null ? (
|
||||||
|
<>
|
||||||
|
<span className="text-gray-300 dark:text-gray-600 select-none">·</span>
|
||||||
|
<span className="text-emerald-600 dark:text-emerald-400">{torrent.seeders} seeds</span>
|
||||||
|
</>
|
||||||
|
) : null}
|
||||||
|
{torrent.score !== undefined && torrent.score !== null ? (
|
||||||
|
<>
|
||||||
|
<span className="text-gray-300 dark:text-gray-600 select-none">·</span>
|
||||||
|
<span className="font-medium">Score {Math.round(torrent.score)}</span>
|
||||||
|
</>
|
||||||
|
) : null}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
|
||||||
{/* Action Buttons */}
|
{/* Action Buttons */}
|
||||||
<div className="border-t border-amber-200 dark:border-amber-800 bg-gray-50 dark:bg-gray-900/50 px-4 py-3 flex gap-2">
|
<div className="border-t border-amber-200 dark:border-amber-800 bg-gray-50 dark:bg-gray-900/50 px-4 py-3 flex gap-2">
|
||||||
<button
|
<button
|
||||||
onClick={() => handleApproveRequest(request.id)}
|
onClick={() => handleApproveRequest(request.id)}
|
||||||
disabled={isLoading}
|
disabled={isLoading}
|
||||||
className="flex-1 inline-flex items-center justify-center gap-2 px-3 py-2 bg-green-600 hover:bg-green-700 disabled:bg-green-400 disabled:cursor-not-allowed text-white text-sm font-medium rounded-lg transition-colors"
|
className="flex-1 inline-flex items-center justify-center gap-1.5 px-3 py-2 bg-green-600 hover:bg-green-700 disabled:bg-green-400 disabled:cursor-not-allowed text-white text-sm font-medium rounded-lg transition-colors"
|
||||||
>
|
>
|
||||||
{isLoading ? (
|
{isLoading ? <LoadingSpinner /> : (
|
||||||
<svg
|
<svg className="w-4 h-4" fill="none" stroke="currentColor" viewBox="0 0 24 24">
|
||||||
className="animate-spin h-4 w-4"
|
<path strokeLinecap="round" strokeLinejoin="round" strokeWidth={2} d="M5 13l4 4L19 7" />
|
||||||
fill="none"
|
|
||||||
viewBox="0 0 24 24"
|
|
||||||
>
|
|
||||||
<circle
|
|
||||||
className="opacity-25"
|
|
||||||
cx="12"
|
|
||||||
cy="12"
|
|
||||||
r="10"
|
|
||||||
stroke="currentColor"
|
|
||||||
strokeWidth="4"
|
|
||||||
/>
|
|
||||||
<path
|
|
||||||
className="opacity-75"
|
|
||||||
fill="currentColor"
|
|
||||||
d="M4 12a8 8 0 018-8V0C5.373 0 0 5.373 0 12h4zm2 5.291A7.962 7.962 0 014 12H0c0 3.042 1.135 5.824 3 7.938l3-2.647z"
|
|
||||||
/>
|
|
||||||
</svg>
|
|
||||||
) : (
|
|
||||||
<svg
|
|
||||||
className="w-4 h-4"
|
|
||||||
fill="none"
|
|
||||||
stroke="currentColor"
|
|
||||||
viewBox="0 0 24 24"
|
|
||||||
>
|
|
||||||
<path
|
|
||||||
strokeLinecap="round"
|
|
||||||
strokeLinejoin="round"
|
|
||||||
strokeWidth={2}
|
|
||||||
d="M5 13l4 4L19 7"
|
|
||||||
/>
|
|
||||||
</svg>
|
</svg>
|
||||||
)}
|
)}
|
||||||
<span>Approve</span>
|
<span>Approve</span>
|
||||||
</button>
|
</button>
|
||||||
|
|
||||||
|
<button
|
||||||
|
onClick={() => setSearchModalRequestId(request.id)}
|
||||||
|
disabled={isLoading}
|
||||||
|
className="flex-1 inline-flex items-center justify-center gap-1.5 px-3 py-2 bg-blue-600 hover:bg-blue-700 disabled:bg-blue-400 disabled:cursor-not-allowed text-white text-sm font-medium rounded-lg transition-colors"
|
||||||
|
>
|
||||||
|
<svg className="w-4 h-4" fill="none" stroke="currentColor" viewBox="0 0 24 24">
|
||||||
|
<path strokeLinecap="round" strokeLinejoin="round" strokeWidth={2} d="M21 21l-6-6m2-5a7 7 0 11-14 0 7 7 0 0114 0z" />
|
||||||
|
</svg>
|
||||||
|
<span>Search</span>
|
||||||
|
</button>
|
||||||
|
|
||||||
<button
|
<button
|
||||||
onClick={() => handleDenyRequest(request.id)}
|
onClick={() => handleDenyRequest(request.id)}
|
||||||
disabled={isLoading}
|
disabled={isLoading}
|
||||||
className="flex-1 inline-flex items-center justify-center gap-2 px-3 py-2 bg-red-600 hover:bg-red-700 disabled:bg-red-400 disabled:cursor-not-allowed text-white text-sm font-medium rounded-lg transition-colors"
|
className="flex-1 inline-flex items-center justify-center gap-1.5 px-3 py-2 bg-red-600 hover:bg-red-700 disabled:bg-red-400 disabled:cursor-not-allowed text-white text-sm font-medium rounded-lg transition-colors"
|
||||||
>
|
>
|
||||||
{isLoading ? (
|
{isLoading ? <LoadingSpinner /> : (
|
||||||
<svg
|
<svg className="w-4 h-4" fill="none" stroke="currentColor" viewBox="0 0 24 24">
|
||||||
className="animate-spin h-4 w-4"
|
<path strokeLinecap="round" strokeLinejoin="round" strokeWidth={2} d="M6 18L18 6M6 6l12 12" />
|
||||||
fill="none"
|
|
||||||
viewBox="0 0 24 24"
|
|
||||||
>
|
|
||||||
<circle
|
|
||||||
className="opacity-25"
|
|
||||||
cx="12"
|
|
||||||
cy="12"
|
|
||||||
r="10"
|
|
||||||
stroke="currentColor"
|
|
||||||
strokeWidth="4"
|
|
||||||
/>
|
|
||||||
<path
|
|
||||||
className="opacity-75"
|
|
||||||
fill="currentColor"
|
|
||||||
d="M4 12a8 8 0 018-8V0C5.373 0 0 5.373 0 12h4zm2 5.291A7.962 7.962 0 014 12H0c0 3.042 1.135 5.824 3 7.938l3-2.647z"
|
|
||||||
/>
|
|
||||||
</svg>
|
|
||||||
) : (
|
|
||||||
<svg
|
|
||||||
className="w-4 h-4"
|
|
||||||
fill="none"
|
|
||||||
stroke="currentColor"
|
|
||||||
viewBox="0 0 24 24"
|
|
||||||
>
|
|
||||||
<path
|
|
||||||
strokeLinecap="round"
|
|
||||||
strokeLinejoin="round"
|
|
||||||
strokeWidth={2}
|
|
||||||
d="M6 18L18 6M6 6l12 12"
|
|
||||||
/>
|
|
||||||
</svg>
|
</svg>
|
||||||
)}
|
)}
|
||||||
<span>Deny</span>
|
<span>Deny</span>
|
||||||
@@ -297,6 +354,26 @@ function PendingApprovalSection({ requests }: { requests: PendingApprovalRequest
|
|||||||
);
|
);
|
||||||
})}
|
})}
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
|
{/* Interactive Search Modal */}
|
||||||
|
{searchModalRequest && (
|
||||||
|
<InteractiveTorrentSearchModal
|
||||||
|
isOpen={!!searchModalRequestId}
|
||||||
|
onClose={() => setSearchModalRequestId(null)}
|
||||||
|
requestId={searchModalRequest.id}
|
||||||
|
audiobook={{
|
||||||
|
title: searchModalRequest.audiobook.title,
|
||||||
|
author: searchModalRequest.audiobook.author,
|
||||||
|
}}
|
||||||
|
searchMode={searchModalRequest.type === 'ebook' ? 'ebook' : 'audiobook'}
|
||||||
|
onConfirm={async (torrent) => {
|
||||||
|
await handleApproveWithTorrent(searchModalRequest.id, torrent);
|
||||||
|
}}
|
||||||
|
onSuccess={() => {
|
||||||
|
setSearchModalRequestId(null);
|
||||||
|
}}
|
||||||
|
/>
|
||||||
|
)}
|
||||||
</div>
|
</div>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -210,6 +210,7 @@ export const getTabValidation = (
|
|||||||
return validated.paths;
|
return validated.paths;
|
||||||
case 'ebook':
|
case 'ebook':
|
||||||
case 'bookdate':
|
case 'bookdate':
|
||||||
|
case 'api':
|
||||||
return true; // These tabs handle their own saving
|
return true; // These tabs handle their own saving
|
||||||
default:
|
default:
|
||||||
return false;
|
return false;
|
||||||
@@ -228,4 +229,5 @@ export const getTabs = (backendMode: 'plex' | 'audiobookshelf') => [
|
|||||||
{ id: 'ebook' as const, label: 'E-book Sidecar', icon: '📖' },
|
{ id: 'ebook' as const, label: 'E-book Sidecar', icon: '📖' },
|
||||||
{ id: 'bookdate' as const, label: 'BookDate', icon: '📚' },
|
{ id: 'bookdate' as const, label: 'BookDate', icon: '📚' },
|
||||||
{ id: 'notifications' as const, label: 'Notifications', icon: '🔔' },
|
{ id: 'notifications' as const, label: 'Notifications', icon: '🔔' },
|
||||||
|
{ id: 'api' as const, label: 'API', icon: '🔑' },
|
||||||
];
|
];
|
||||||
|
|||||||
@@ -100,6 +100,8 @@ export interface PathsSettings {
|
|||||||
ebookPathTemplate?: string;
|
ebookPathTemplate?: string;
|
||||||
metadataTaggingEnabled: boolean;
|
metadataTaggingEnabled: boolean;
|
||||||
chapterMergingEnabled: boolean;
|
chapterMergingEnabled: boolean;
|
||||||
|
fileRenameEnabled: boolean;
|
||||||
|
fileRenameTemplate?: string;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -241,4 +243,4 @@ export interface BookDateModel {
|
|||||||
/**
|
/**
|
||||||
* Tab identifier type
|
* Tab identifier type
|
||||||
*/
|
*/
|
||||||
export type SettingsTab = 'library' | 'auth' | 'prowlarr' | 'download' | 'paths' | 'ebook' | 'bookdate' | 'notifications';
|
export type SettingsTab = 'library' | 'auth' | 'prowlarr' | 'download' | 'paths' | 'ebook' | 'bookdate' | 'notifications' | 'api';
|
||||||
|
|||||||
@@ -23,6 +23,7 @@ import { PathsTab } from './tabs/PathsTab/PathsTab';
|
|||||||
import { EbookTab } from './tabs/EbookTab/EbookTab';
|
import { EbookTab } from './tabs/EbookTab/EbookTab';
|
||||||
import { BookDateTab } from './tabs/BookDateTab/BookDateTab';
|
import { BookDateTab } from './tabs/BookDateTab/BookDateTab';
|
||||||
import { NotificationsTab } from './tabs/NotificationsTab';
|
import { NotificationsTab } from './tabs/NotificationsTab';
|
||||||
|
import { ApiTab } from './tabs/ApiTab/ApiTab';
|
||||||
|
|
||||||
// Types and Helpers
|
// Types and Helpers
|
||||||
import type { Settings, SettingsTab, IndexerConfig, SavedIndexerConfig, Message } from './lib/types';
|
import type { Settings, SettingsTab, IndexerConfig, SavedIndexerConfig, Message } from './lib/types';
|
||||||
@@ -346,8 +347,11 @@ export default function AdminSettings() {
|
|||||||
{/* Notifications Tab */}
|
{/* Notifications Tab */}
|
||||||
{activeTab === 'notifications' && <NotificationsTab />}
|
{activeTab === 'notifications' && <NotificationsTab />}
|
||||||
|
|
||||||
|
{/* API Tab */}
|
||||||
|
{activeTab === 'api' && <ApiTab />}
|
||||||
|
|
||||||
{/* Save Button (only for tabs that save through main page) */}
|
{/* Save Button (only for tabs that save through main page) */}
|
||||||
{activeTab !== 'ebook' && activeTab !== 'bookdate' && activeTab !== 'notifications' && (
|
{activeTab !== 'ebook' && activeTab !== 'bookdate' && activeTab !== 'notifications' && activeTab !== 'api' && (
|
||||||
<div className="mt-8 flex gap-4">
|
<div className="mt-8 flex gap-4">
|
||||||
<Button
|
<Button
|
||||||
onClick={saveSettings}
|
onClick={saveSettings}
|
||||||
|
|||||||
@@ -0,0 +1,307 @@
|
|||||||
|
/**
|
||||||
|
* Component: API Token Management Tab (Admin)
|
||||||
|
* Documentation: documentation/backend/services/api-tokens.md
|
||||||
|
*/
|
||||||
|
|
||||||
|
'use client';
|
||||||
|
|
||||||
|
import { useState, useEffect, useCallback } from 'react';
|
||||||
|
import { fetchWithAuth } from '@/lib/utils/api';
|
||||||
|
import { ConfirmDialog } from '@/app/admin/components/ConfirmDialog';
|
||||||
|
import { useApiTokens } from '@/lib/hooks/useApiTokens';
|
||||||
|
import { getInstanceUrl } from '@/lib/utils/client-url';
|
||||||
|
import Link from 'next/link';
|
||||||
|
import type { AdminApiToken } from '@/lib/types/api-tokens';
|
||||||
|
|
||||||
|
interface UserOption {
|
||||||
|
id: string;
|
||||||
|
plexUsername: string;
|
||||||
|
role: string;
|
||||||
|
}
|
||||||
|
|
||||||
|
export function ApiTab() {
|
||||||
|
const api = useApiTokens<AdminApiToken>({ basePath: '/api/admin/api-tokens' });
|
||||||
|
|
||||||
|
// Admin-specific state
|
||||||
|
const [users, setUsers] = useState<UserOption[]>([]);
|
||||||
|
const [newTokenUserId, setNewTokenUserId] = useState('');
|
||||||
|
|
||||||
|
const fetchUsers = useCallback(async () => {
|
||||||
|
try {
|
||||||
|
const response = await fetchWithAuth('/api/admin/users');
|
||||||
|
if (response.ok) {
|
||||||
|
const data = await response.json();
|
||||||
|
setUsers(data.users.map((u: any) => ({ id: u.id, plexUsername: u.plexUsername, role: u.role })));
|
||||||
|
}
|
||||||
|
} catch {
|
||||||
|
// Non-critical, user selector just won't populate
|
||||||
|
}
|
||||||
|
}, []);
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
fetchUsers();
|
||||||
|
}, [fetchUsers]);
|
||||||
|
|
||||||
|
const handleCreate = async () => {
|
||||||
|
const extraBody: Record<string, string> = {};
|
||||||
|
if (newTokenUserId) extraBody.userId = newTokenUserId;
|
||||||
|
const created = await api.handleCreate(extraBody);
|
||||||
|
// Reset admin-specific fields only when create succeeds
|
||||||
|
if (created) {
|
||||||
|
setNewTokenUserId('');
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
const handleCancel = () => {
|
||||||
|
api.resetForm();
|
||||||
|
setNewTokenUserId('');
|
||||||
|
};
|
||||||
|
|
||||||
|
if (api.loading) {
|
||||||
|
return (
|
||||||
|
<div className="flex items-center justify-center py-12">
|
||||||
|
<div className="animate-spin rounded-full h-8 w-8 border-b-2 border-blue-600"></div>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div className="space-y-6">
|
||||||
|
<div>
|
||||||
|
<h2 className="text-xl font-semibold text-gray-900 dark:text-gray-100">API Tokens</h2>
|
||||||
|
<p className="text-sm text-gray-500 dark:text-gray-400 mt-1">
|
||||||
|
Manage API tokens for all users. Create tokens for any user for programmatic access.{' '}
|
||||||
|
<Link href="/api-docs" className="text-blue-600 dark:text-blue-400 hover:underline">
|
||||||
|
View API documentation
|
||||||
|
</Link>
|
||||||
|
</p>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
{/* Error display */}
|
||||||
|
{api.error && (
|
||||||
|
<div className="p-3 rounded-lg bg-red-50 dark:bg-red-900/20 text-red-800 dark:text-red-200 text-sm">
|
||||||
|
{api.error}
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
|
||||||
|
{/* Newly created token banner */}
|
||||||
|
{api.createdToken && (
|
||||||
|
<div className="p-4 rounded-lg bg-green-50 dark:bg-green-900/20 border border-green-200 dark:border-green-800">
|
||||||
|
<div className="flex items-start gap-3">
|
||||||
|
<svg className="w-5 h-5 text-green-600 dark:text-green-400 mt-0.5 flex-shrink-0" fill="none" stroke="currentColor" viewBox="0 0 24 24">
|
||||||
|
<path strokeLinecap="round" strokeLinejoin="round" strokeWidth={2} d="M9 12l2 2 4-4m6 2a9 9 0 11-18 0 9 9 0 0118 0z" />
|
||||||
|
</svg>
|
||||||
|
<div className="flex-1 min-w-0">
|
||||||
|
<p className="text-sm font-medium text-green-800 dark:text-green-200">
|
||||||
|
Token created successfully! Copy it now — it won't be shown again.
|
||||||
|
</p>
|
||||||
|
<div className="mt-2 flex items-center gap-2">
|
||||||
|
<code className="flex-1 text-sm bg-white dark:bg-gray-900 px-3 py-2 rounded border border-green-300 dark:border-green-700 text-gray-900 dark:text-gray-100 font-mono break-all">
|
||||||
|
{api.createdToken}
|
||||||
|
</code>
|
||||||
|
<button
|
||||||
|
onClick={api.handleCopy}
|
||||||
|
className="flex-shrink-0 px-3 py-2 text-sm font-medium rounded-lg bg-green-600 hover:bg-green-700 text-white transition-colors"
|
||||||
|
>
|
||||||
|
{api.copied ? 'Copied!' : 'Copy'}
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<button
|
||||||
|
type="button"
|
||||||
|
aria-label="Dismiss token banner"
|
||||||
|
onClick={api.dismissCreatedToken}
|
||||||
|
className="flex-shrink-0 text-green-600 dark:text-green-400 hover:text-green-800 dark:hover:text-green-200"
|
||||||
|
>
|
||||||
|
<svg className="w-5 h-5" fill="none" stroke="currentColor" viewBox="0 0 24 24">
|
||||||
|
<path strokeLinecap="round" strokeLinejoin="round" strokeWidth={2} d="M6 18L18 6M6 6l12 12" />
|
||||||
|
</svg>
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
|
||||||
|
{/* Create token form */}
|
||||||
|
{api.showCreateForm ? (
|
||||||
|
<div className="p-4 rounded-lg border border-gray-200 dark:border-gray-700 bg-gray-50 dark:bg-gray-900/50 space-y-4">
|
||||||
|
<h3 className="text-sm font-medium text-gray-900 dark:text-gray-100">Create New Token</h3>
|
||||||
|
<div className="grid grid-cols-1 sm:grid-cols-2 gap-4">
|
||||||
|
<div>
|
||||||
|
<label className="block text-sm font-medium text-gray-700 dark:text-gray-300 mb-1">
|
||||||
|
Name
|
||||||
|
</label>
|
||||||
|
<input
|
||||||
|
type="text"
|
||||||
|
value={api.newTokenName}
|
||||||
|
onChange={(e) => api.setNewTokenName(e.target.value)}
|
||||||
|
placeholder="e.g., Home Assistant, Webhook"
|
||||||
|
className="w-full rounded-lg border border-gray-300 dark:border-gray-600 bg-white dark:bg-gray-800 px-3 py-2 text-sm text-gray-900 dark:text-gray-100 focus:border-blue-500 focus:ring-1 focus:ring-blue-500"
|
||||||
|
onKeyDown={(e) => e.key === 'Enter' && handleCreate()}
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
<div>
|
||||||
|
<label className="block text-sm font-medium text-gray-700 dark:text-gray-300 mb-1">
|
||||||
|
Expiration
|
||||||
|
</label>
|
||||||
|
<select
|
||||||
|
value={api.newTokenExpiry}
|
||||||
|
onChange={(e) => api.setNewTokenExpiry(e.target.value)}
|
||||||
|
className="w-full rounded-lg border border-gray-300 dark:border-gray-600 bg-white dark:bg-gray-800 px-3 py-2 text-sm text-gray-900 dark:text-gray-100 focus:border-blue-500 focus:ring-1 focus:ring-blue-500"
|
||||||
|
>
|
||||||
|
<option value="never">Never</option>
|
||||||
|
<option value="30d">30 days</option>
|
||||||
|
<option value="90d">90 days</option>
|
||||||
|
<option value="1y">1 year</option>
|
||||||
|
</select>
|
||||||
|
</div>
|
||||||
|
<div>
|
||||||
|
<label className="block text-sm font-medium text-gray-700 dark:text-gray-300 mb-1">
|
||||||
|
User (acts as)
|
||||||
|
</label>
|
||||||
|
<select
|
||||||
|
value={newTokenUserId}
|
||||||
|
onChange={(e) => setNewTokenUserId(e.target.value)}
|
||||||
|
className="w-full rounded-lg border border-gray-300 dark:border-gray-600 bg-white dark:bg-gray-800 px-3 py-2 text-sm text-gray-900 dark:text-gray-100 focus:border-blue-500 focus:ring-1 focus:ring-blue-500"
|
||||||
|
>
|
||||||
|
<option value="">Current user (default)</option>
|
||||||
|
{users.map((u) => (
|
||||||
|
<option key={u.id} value={u.id}>
|
||||||
|
{u.plexUsername} ({u.role})
|
||||||
|
</option>
|
||||||
|
))}
|
||||||
|
</select>
|
||||||
|
<p className="mt-1 text-xs text-gray-500 dark:text-gray-400">
|
||||||
|
Token will inherit the selected user's role
|
||||||
|
</p>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div className="flex gap-2">
|
||||||
|
<button
|
||||||
|
onClick={handleCreate}
|
||||||
|
disabled={api.creating || !api.newTokenName.trim()}
|
||||||
|
className="px-4 py-2 text-sm font-medium rounded-lg bg-blue-600 hover:bg-blue-700 disabled:bg-blue-400 text-white transition-colors"
|
||||||
|
>
|
||||||
|
{api.creating ? 'Creating...' : 'Create Token'}
|
||||||
|
</button>
|
||||||
|
<button
|
||||||
|
onClick={handleCancel}
|
||||||
|
className="px-4 py-2 text-sm font-medium rounded-lg bg-gray-200 dark:bg-gray-700 hover:bg-gray-300 dark:hover:bg-gray-600 text-gray-900 dark:text-gray-100 transition-colors"
|
||||||
|
>
|
||||||
|
Cancel
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
) : (
|
||||||
|
<button
|
||||||
|
onClick={() => api.setShowCreateForm(true)}
|
||||||
|
className="px-4 py-2 text-sm font-medium rounded-lg bg-blue-600 hover:bg-blue-700 text-white transition-colors"
|
||||||
|
>
|
||||||
|
Create New Token
|
||||||
|
</button>
|
||||||
|
)}
|
||||||
|
|
||||||
|
{/* Token list */}
|
||||||
|
{api.tokens.length === 0 ? (
|
||||||
|
<div className="text-center py-8 text-gray-500 dark:text-gray-400">
|
||||||
|
<svg className="mx-auto h-12 w-12 text-gray-400" fill="none" stroke="currentColor" viewBox="0 0 24 24">
|
||||||
|
<path strokeLinecap="round" strokeLinejoin="round" strokeWidth={1.5} d="M15 7a2 2 0 012 2m4 0a6 6 0 01-7.743 5.743L11 17H9v2H7v2H4a1 1 0 01-1-1v-2.586a1 1 0 01.293-.707l5.964-5.964A6 6 0 1121 9z" />
|
||||||
|
</svg>
|
||||||
|
<p className="mt-2 text-sm">No API tokens yet</p>
|
||||||
|
<p className="text-xs mt-1">Create a token to enable programmatic API access</p>
|
||||||
|
</div>
|
||||||
|
) : (
|
||||||
|
<div className="overflow-x-auto">
|
||||||
|
<table className="w-full text-sm">
|
||||||
|
<thead>
|
||||||
|
<tr className="border-b border-gray-200 dark:border-gray-700">
|
||||||
|
<th className="text-left py-3 px-2 font-medium text-gray-500 dark:text-gray-400">Name</th>
|
||||||
|
<th className="text-left py-3 px-2 font-medium text-gray-500 dark:text-gray-400">Token</th>
|
||||||
|
<th className="text-left py-3 px-2 font-medium text-gray-500 dark:text-gray-400">Acts As</th>
|
||||||
|
<th className="text-left py-3 px-2 font-medium text-gray-500 dark:text-gray-400">Role</th>
|
||||||
|
<th className="text-left py-3 px-2 font-medium text-gray-500 dark:text-gray-400">Created By</th>
|
||||||
|
<th className="text-left py-3 px-2 font-medium text-gray-500 dark:text-gray-400">Last Used</th>
|
||||||
|
<th className="text-left py-3 px-2 font-medium text-gray-500 dark:text-gray-400">Expires</th>
|
||||||
|
<th className="text-right py-3 px-2 font-medium text-gray-500 dark:text-gray-400">Actions</th>
|
||||||
|
</tr>
|
||||||
|
</thead>
|
||||||
|
<tbody>
|
||||||
|
{api.tokens.map((token) => (
|
||||||
|
<tr key={token.id} className="border-b border-gray-100 dark:border-gray-800">
|
||||||
|
<td className="py-3 px-2 text-gray-900 dark:text-gray-100 font-medium">{token.name}</td>
|
||||||
|
<td className="py-3 px-2">
|
||||||
|
<code className="text-xs bg-gray-100 dark:bg-gray-900 px-2 py-1 rounded text-gray-600 dark:text-gray-400 font-mono">
|
||||||
|
{token.tokenPrefix}...
|
||||||
|
</code>
|
||||||
|
</td>
|
||||||
|
<td className="py-3 px-2 text-gray-500 dark:text-gray-400">{token.tokenUser}</td>
|
||||||
|
<td className="py-3 px-2">
|
||||||
|
<span className={`inline-flex items-center px-2 py-0.5 rounded-full text-xs font-medium ${
|
||||||
|
token.role === 'admin'
|
||||||
|
? 'bg-purple-100 dark:bg-purple-900/30 text-purple-700 dark:text-purple-300'
|
||||||
|
: 'bg-gray-100 dark:bg-gray-700 text-gray-600 dark:text-gray-400'
|
||||||
|
}`}>
|
||||||
|
{token.role}
|
||||||
|
</span>
|
||||||
|
</td>
|
||||||
|
<td className="py-3 px-2 text-gray-500 dark:text-gray-400">{token.createdBy}</td>
|
||||||
|
<td className="py-3 px-2 text-gray-500 dark:text-gray-400">{api.formatDate(token.lastUsedAt)}</td>
|
||||||
|
<td className="py-3 px-2 text-gray-500 dark:text-gray-400">
|
||||||
|
{token.expiresAt ? (
|
||||||
|
<span className={new Date(token.expiresAt) < new Date() ? 'text-red-500' : ''}>
|
||||||
|
{api.formatDate(token.expiresAt)}
|
||||||
|
{new Date(token.expiresAt) < new Date() && ' (expired)'}
|
||||||
|
</span>
|
||||||
|
) : (
|
||||||
|
'Never'
|
||||||
|
)}
|
||||||
|
</td>
|
||||||
|
<td className="py-3 px-2 text-right">
|
||||||
|
<button
|
||||||
|
onClick={() => api.setConfirmRevokeId(token.id)}
|
||||||
|
disabled={api.deletingId === token.id}
|
||||||
|
className="px-3 py-1 text-xs font-medium rounded-lg bg-red-100 dark:bg-red-900/30 hover:bg-red-200 dark:hover:bg-red-900/50 text-red-700 dark:text-red-300 transition-colors disabled:opacity-50"
|
||||||
|
>
|
||||||
|
{api.deletingId === token.id ? 'Revoking...' : 'Revoke'}
|
||||||
|
</button>
|
||||||
|
</td>
|
||||||
|
</tr>
|
||||||
|
))}
|
||||||
|
</tbody>
|
||||||
|
</table>
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
|
||||||
|
{/* Usage instructions */}
|
||||||
|
<div className="p-4 rounded-lg bg-gray-50 dark:bg-gray-900/50 border border-gray-200 dark:border-gray-700">
|
||||||
|
<h3 className="text-sm font-medium text-gray-900 dark:text-gray-100 mb-2">Usage</h3>
|
||||||
|
<p className="text-sm text-gray-600 dark:text-gray-400 mb-2">
|
||||||
|
Include the token in the <code className="px-1 py-0.5 bg-gray-200 dark:bg-gray-800 rounded text-xs">Authorization</code> header:
|
||||||
|
</p>
|
||||||
|
<pre className="text-xs bg-gray-900 dark:bg-black text-gray-100 p-3 rounded-lg overflow-x-auto">
|
||||||
|
{`curl -H "Authorization: Bearer rmab_your_token_here" \\
|
||||||
|
${getInstanceUrl()}/api/requests`}
|
||||||
|
</pre>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
{/* Revoke confirmation dialog */}
|
||||||
|
<ConfirmDialog
|
||||||
|
isOpen={api.confirmRevokeId !== null}
|
||||||
|
title="Revoke API token"
|
||||||
|
message={
|
||||||
|
<>
|
||||||
|
Are you sure you want to revoke{' '}
|
||||||
|
<span className="font-medium text-gray-700 dark:text-gray-200">
|
||||||
|
“{api.tokens.find((t) => t.id === api.confirmRevokeId)?.name ?? 'this token'}”
|
||||||
|
</span>
|
||||||
|
? Any integrations using this token will immediately lose access. This cannot be undone.
|
||||||
|
</>
|
||||||
|
}
|
||||||
|
confirmLabel="Revoke token"
|
||||||
|
cancelLabel="Cancel"
|
||||||
|
confirmVariant="danger"
|
||||||
|
onConfirm={api.handleDeleteConfirmed}
|
||||||
|
onCancel={() => api.setConfirmRevokeId(null)}
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
}
|
||||||
@@ -90,6 +90,7 @@ export function BookDateTab({ onSuccess, onError }: BookDateTabProps) {
|
|||||||
>
|
>
|
||||||
<option value="openai">OpenAI</option>
|
<option value="openai">OpenAI</option>
|
||||||
<option value="claude">Claude (Anthropic)</option>
|
<option value="claude">Claude (Anthropic)</option>
|
||||||
|
<option value="gemini">Google Gemini</option>
|
||||||
<option value="custom">Custom (OpenAI-compatible)</option>
|
<option value="custom">Custom (OpenAI-compatible)</option>
|
||||||
</select>
|
</select>
|
||||||
</div>
|
</div>
|
||||||
@@ -136,7 +137,7 @@ export function BookDateTab({ onSuccess, onError }: BookDateTabProps) {
|
|||||||
? 'Leave blank for local models'
|
? 'Leave blank for local models'
|
||||||
: configured
|
: configured
|
||||||
? '••••••••••••••••'
|
? '••••••••••••••••'
|
||||||
: (provider === 'openai' ? 'sk-...' : 'sk-ant-...')
|
: (provider === 'openai' ? 'sk-...' : provider === 'gemini' ? 'AIza...' : 'sk-ant-...')
|
||||||
}
|
}
|
||||||
/>
|
/>
|
||||||
<p className="text-sm text-gray-500 dark:text-gray-400 mt-1">
|
<p className="text-sm text-gray-500 dark:text-gray-400 mt-1">
|
||||||
|
|||||||
@@ -90,9 +90,9 @@ export function EbookTab({ ebook, onChange, onSuccess, onError, markAsSaved }: E
|
|||||||
</label>
|
</label>
|
||||||
<Input
|
<Input
|
||||||
type="text"
|
type="text"
|
||||||
value={ebook.baseUrl || 'https://annas-archive.li'}
|
value={ebook.baseUrl || 'https://annas-archive.gl'}
|
||||||
onChange={(e) => updateEbook('baseUrl', e.target.value)}
|
onChange={(e) => updateEbook('baseUrl', e.target.value)}
|
||||||
placeholder="https://annas-archive.li"
|
placeholder="https://annas-archive.gl"
|
||||||
className="font-mono"
|
className="font-mono"
|
||||||
/>
|
/>
|
||||||
<p className="text-sm text-gray-500 dark:text-gray-400 mt-1">
|
<p className="text-sm text-gray-500 dark:text-gray-400 mt-1">
|
||||||
|
|||||||
@@ -53,7 +53,7 @@ export function useEbookSettings({ ebook, onChange, onSuccess, onError, markAsSa
|
|||||||
headers: { 'Content-Type': 'application/json' },
|
headers: { 'Content-Type': 'application/json' },
|
||||||
body: JSON.stringify({
|
body: JSON.stringify({
|
||||||
url: ebook.flaresolverrUrl,
|
url: ebook.flaresolverrUrl,
|
||||||
baseUrl: ebook.baseUrl || 'https://annas-archive.li',
|
baseUrl: ebook.baseUrl || 'https://annas-archive.gl',
|
||||||
}),
|
}),
|
||||||
});
|
});
|
||||||
|
|
||||||
@@ -83,7 +83,7 @@ export function useEbookSettings({ ebook, onChange, onSuccess, onError, markAsSa
|
|||||||
annasArchiveEnabled: ebook.annasArchiveEnabled || false,
|
annasArchiveEnabled: ebook.annasArchiveEnabled || false,
|
||||||
indexerSearchEnabled: ebook.indexerSearchEnabled || false,
|
indexerSearchEnabled: ebook.indexerSearchEnabled || false,
|
||||||
format: ebook.preferredFormat || 'epub',
|
format: ebook.preferredFormat || 'epub',
|
||||||
baseUrl: ebook.baseUrl || 'https://annas-archive.li',
|
baseUrl: ebook.baseUrl || 'https://annas-archive.gl',
|
||||||
flaresolverrUrl: ebook.flaresolverrUrl || '',
|
flaresolverrUrl: ebook.flaresolverrUrl || '',
|
||||||
autoGrabEnabled: ebook.autoGrabEnabled ?? true,
|
autoGrabEnabled: ebook.autoGrabEnabled ?? true,
|
||||||
kindleFixEnabled: ebook.kindleFixEnabled ?? false,
|
kindleFixEnabled: ebook.kindleFixEnabled ?? false,
|
||||||
|
|||||||
@@ -10,7 +10,7 @@ import { Button } from '@/components/ui/Button';
|
|||||||
import { Input } from '@/components/ui/Input';
|
import { Input } from '@/components/ui/Input';
|
||||||
import { usePathsSettings } from './usePathsSettings';
|
import { usePathsSettings } from './usePathsSettings';
|
||||||
import type { PathsSettings } from '../../lib/types';
|
import type { PathsSettings } from '../../lib/types';
|
||||||
import { validateTemplate, generateMockPreviews } from '@/lib/utils/path-template.util';
|
import { validateTemplate, generateMockPreviews, validateFilenameTemplate, generateMockFilenamePreviews } from '@/lib/utils/path-template.util';
|
||||||
|
|
||||||
interface PathsTabProps {
|
interface PathsTabProps {
|
||||||
paths: PathsSettings;
|
paths: PathsSettings;
|
||||||
@@ -24,6 +24,13 @@ interface TemplatePreview {
|
|||||||
previewPaths?: string[];
|
previewPaths?: string[];
|
||||||
}
|
}
|
||||||
|
|
||||||
|
interface FilenamePreview {
|
||||||
|
isValid: boolean;
|
||||||
|
error?: string;
|
||||||
|
single?: string[];
|
||||||
|
multi?: string[];
|
||||||
|
}
|
||||||
|
|
||||||
export function PathsTab({ paths, onChange, onValidationChange }: PathsTabProps) {
|
export function PathsTab({ paths, onChange, onValidationChange }: PathsTabProps) {
|
||||||
const { testing, testResult, updatePath, testPaths } = usePathsSettings({
|
const { testing, testResult, updatePath, testPaths } = usePathsSettings({
|
||||||
paths,
|
paths,
|
||||||
@@ -73,6 +80,34 @@ export function PathsTab({ paths, onChange, onValidationChange }: PathsTabProps)
|
|||||||
}
|
}
|
||||||
}, [paths.ebookPathTemplate]);
|
}, [paths.ebookPathTemplate]);
|
||||||
|
|
||||||
|
// Live preview state for filename template
|
||||||
|
const [filenamePreview, setFilenamePreview] = useState<FilenamePreview | null>(null);
|
||||||
|
|
||||||
|
// Update filename live preview whenever template changes
|
||||||
|
useEffect(() => {
|
||||||
|
if (!paths.fileRenameEnabled) {
|
||||||
|
setFilenamePreview(null);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
const template = paths.fileRenameTemplate || '{title}';
|
||||||
|
const validation = validateFilenameTemplate(template);
|
||||||
|
|
||||||
|
if (validation.valid) {
|
||||||
|
const previews = generateMockFilenamePreviews(template);
|
||||||
|
setFilenamePreview({
|
||||||
|
isValid: true,
|
||||||
|
single: previews.single,
|
||||||
|
multi: previews.multi,
|
||||||
|
});
|
||||||
|
} else {
|
||||||
|
setFilenamePreview({
|
||||||
|
isValid: false,
|
||||||
|
error: validation.error,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}, [paths.fileRenameTemplate, paths.fileRenameEnabled]);
|
||||||
|
|
||||||
const audiobookTemplate = paths.audiobookPathTemplate || '{author}/{title} {asin}';
|
const audiobookTemplate = paths.audiobookPathTemplate || '{author}/{title} {asin}';
|
||||||
const ebookTemplate = paths.ebookPathTemplate || '{author}/{title} {asin}';
|
const ebookTemplate = paths.ebookPathTemplate || '{author}/{title} {asin}';
|
||||||
const ebookMatchesAudiobook = ebookTemplate === audiobookTemplate;
|
const ebookMatchesAudiobook = ebookTemplate === audiobookTemplate;
|
||||||
@@ -218,6 +253,83 @@ export function PathsTab({ paths, onChange, onValidationChange }: PathsTabProps)
|
|||||||
)}
|
)}
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
|
{/* File Rename Toggle */}
|
||||||
|
<div className="bg-gray-50 dark:bg-gray-800 rounded-lg p-4 border border-gray-200 dark:border-gray-700">
|
||||||
|
<div className="flex items-start gap-4">
|
||||||
|
<input
|
||||||
|
type="checkbox"
|
||||||
|
id="file-rename-settings"
|
||||||
|
checked={paths.fileRenameEnabled}
|
||||||
|
onChange={(e) => updatePath('fileRenameEnabled', e.target.checked)}
|
||||||
|
className="mt-1 h-5 w-5 rounded border-gray-300 text-blue-600 focus:ring-blue-500"
|
||||||
|
/>
|
||||||
|
<div className="flex-1">
|
||||||
|
<label
|
||||||
|
htmlFor="file-rename-settings"
|
||||||
|
className="block text-sm font-medium text-gray-900 dark:text-gray-100 cursor-pointer"
|
||||||
|
>
|
||||||
|
Rename files during organization
|
||||||
|
</label>
|
||||||
|
<p className="text-sm text-gray-600 dark:text-gray-400 mt-1">
|
||||||
|
Rename audio and ebook files using a custom naming template when organizing into the media
|
||||||
|
library. When multiple files exist (e.g. chapterized MP3s), an index number is appended.
|
||||||
|
</p>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
{/* File Naming Template (shown when enabled) */}
|
||||||
|
{paths.fileRenameEnabled && (
|
||||||
|
<div className="mt-4 pl-9">
|
||||||
|
<label className="block text-sm font-medium text-gray-700 dark:text-gray-300 mb-2">
|
||||||
|
File Naming Template
|
||||||
|
</label>
|
||||||
|
<Input
|
||||||
|
type="text"
|
||||||
|
value={paths.fileRenameTemplate || '{title}'}
|
||||||
|
onChange={(e) => updatePath('fileRenameTemplate', e.target.value)}
|
||||||
|
placeholder="{title}"
|
||||||
|
className="font-mono"
|
||||||
|
/>
|
||||||
|
<p className="text-sm text-gray-500 dark:text-gray-400 mt-1">
|
||||||
|
Uses the same variables as the organization template. Do not include the file extension.
|
||||||
|
</p>
|
||||||
|
|
||||||
|
{/* Filename Validation Error */}
|
||||||
|
{filenamePreview && !filenamePreview.isValid && (
|
||||||
|
<div className="mt-3 p-3 rounded-lg text-sm flex items-start gap-2 bg-red-50 dark:bg-red-900/20 border border-red-200 dark:border-red-800 text-red-800 dark:text-red-200">
|
||||||
|
<span className="flex-shrink-0 mt-0.5">✗</span>
|
||||||
|
<div className="flex-1">
|
||||||
|
<span>{filenamePreview.error || 'Invalid filename template'}</span>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
|
||||||
|
{/* Filename Preview */}
|
||||||
|
{filenamePreview && filenamePreview.isValid && (
|
||||||
|
<div className="mt-3 bg-gray-50 dark:bg-gray-800 border border-gray-200 dark:border-gray-700 rounded-lg p-4">
|
||||||
|
<h4 className="text-sm font-semibold text-gray-900 dark:text-gray-100 mb-2">
|
||||||
|
Single File
|
||||||
|
</h4>
|
||||||
|
<div className="space-y-1.5 text-sm font-mono text-gray-700 dark:text-gray-300">
|
||||||
|
{filenamePreview.single?.map((preview, index) => (
|
||||||
|
<div key={index} className="text-xs">{preview}</div>
|
||||||
|
))}
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<h4 className="text-sm font-semibold text-gray-900 dark:text-gray-100 mt-3 mb-2">
|
||||||
|
Multiple Files (chapterized)
|
||||||
|
</h4>
|
||||||
|
<div className="space-y-1.5 text-sm font-mono text-gray-700 dark:text-gray-300">
|
||||||
|
{filenamePreview.multi?.map((preview, index) => (
|
||||||
|
<div key={index} className="text-xs">{preview}</div>
|
||||||
|
))}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
|
||||||
{/* Variable Reference Panel (shared for both templates) */}
|
{/* Variable Reference Panel (shared for both templates) */}
|
||||||
<div className="bg-blue-50 dark:bg-blue-900/20 border border-blue-200 dark:border-blue-800 rounded-lg p-4">
|
<div className="bg-blue-50 dark:bg-blue-900/20 border border-blue-200 dark:border-blue-800 rounded-lg p-4">
|
||||||
<h3 className="text-sm font-semibold text-blue-900 dark:text-blue-100 mb-3">
|
<h3 className="text-sm font-semibold text-blue-900 dark:text-blue-100 mb-3">
|
||||||
@@ -255,6 +367,27 @@ export function PathsTab({ paths, onChange, onValidationChange }: PathsTabProps)
|
|||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
|
{/* Conditional Syntax Help */}
|
||||||
|
<div className="bg-amber-50 dark:bg-amber-900/20 border border-amber-200 dark:border-amber-800 rounded-lg p-4">
|
||||||
|
<h3 className="text-sm font-semibold text-amber-900 dark:text-amber-100 mb-2">
|
||||||
|
Conditional Syntax
|
||||||
|
</h3>
|
||||||
|
<p className="text-sm text-gray-600 dark:text-gray-400 mb-2">
|
||||||
|
Wrap text around a variable in <code className="text-amber-700 dark:text-amber-300 font-mono">{'{ }'}</code> to
|
||||||
|
include that text only when the variable has a value. If the variable is empty, the entire block is removed.
|
||||||
|
</p>
|
||||||
|
<div className="text-sm font-mono bg-white dark:bg-gray-900 rounded px-3 py-2 border border-amber-100 dark:border-amber-900">
|
||||||
|
<div className="text-gray-700 dark:text-gray-300">
|
||||||
|
<code className="text-amber-700 dark:text-amber-300">{'{Book seriesPart - }'}</code>
|
||||||
|
</div>
|
||||||
|
<div className="text-xs text-gray-500 dark:text-gray-400 mt-1">
|
||||||
|
With value: <span className="text-green-700 dark:text-green-400">Book 1 - </span>
|
||||||
|
•
|
||||||
|
Without value: <span className="text-red-700 dark:text-red-400">(removed)</span>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
{/* Metadata Tagging Toggle */}
|
{/* Metadata Tagging Toggle */}
|
||||||
<div className="bg-gray-50 dark:bg-gray-800 rounded-lg p-4 border border-gray-200 dark:border-gray-700">
|
<div className="bg-gray-50 dark:bg-gray-800 rounded-lg p-4 border border-gray-200 dark:border-gray-700">
|
||||||
<div className="flex items-start gap-4">
|
<div className="flex items-start gap-4">
|
||||||
|
|||||||
@@ -28,6 +28,7 @@ interface User {
|
|||||||
lastLoginAt: string | null;
|
lastLoginAt: string | null;
|
||||||
autoApproveRequests: boolean | null;
|
autoApproveRequests: boolean | null;
|
||||||
interactiveSearchAccess: boolean | null;
|
interactiveSearchAccess: boolean | null;
|
||||||
|
downloadAccess: boolean | null;
|
||||||
_count: {
|
_count: {
|
||||||
requests: number;
|
requests: number;
|
||||||
};
|
};
|
||||||
@@ -193,6 +194,10 @@ function AdminUsersPageContent() {
|
|||||||
'/api/admin/settings/interactive-search',
|
'/api/admin/settings/interactive-search',
|
||||||
authenticatedFetcher
|
authenticatedFetcher
|
||||||
);
|
);
|
||||||
|
const { data: globalDownloadAccessData, mutate: mutateGlobalDownloadAccess } = useSWR(
|
||||||
|
'/api/admin/settings/download-access',
|
||||||
|
authenticatedFetcher
|
||||||
|
);
|
||||||
const [editDialog, setEditDialog] = useState<{
|
const [editDialog, setEditDialog] = useState<{
|
||||||
isOpen: boolean;
|
isOpen: boolean;
|
||||||
user: User | null;
|
user: User | null;
|
||||||
@@ -212,6 +217,7 @@ function AdminUsersPageContent() {
|
|||||||
const [deleting, setDeleting] = useState(false);
|
const [deleting, setDeleting] = useState(false);
|
||||||
const [globalAutoApprove, setGlobalAutoApprove] = useState<boolean>(false);
|
const [globalAutoApprove, setGlobalAutoApprove] = useState<boolean>(false);
|
||||||
const [globalInteractiveSearch, setGlobalInteractiveSearch] = useState<boolean>(true);
|
const [globalInteractiveSearch, setGlobalInteractiveSearch] = useState<boolean>(true);
|
||||||
|
const [globalDownloadAccess, setGlobalDownloadAccess] = useState<boolean>(true);
|
||||||
const [globalSettingsOpen, setGlobalSettingsOpen] = useState(false);
|
const [globalSettingsOpen, setGlobalSettingsOpen] = useState(false);
|
||||||
const [permissionsUserId, setPermissionsUserId] = useState<string | null>(null);
|
const [permissionsUserId, setPermissionsUserId] = useState<string | null>(null);
|
||||||
const toast = useToast();
|
const toast = useToast();
|
||||||
@@ -237,6 +243,15 @@ function AdminUsersPageContent() {
|
|||||||
}
|
}
|
||||||
}, [globalInteractiveSearchData]);
|
}, [globalInteractiveSearchData]);
|
||||||
|
|
||||||
|
// Sync global download access state (default to true if not set)
|
||||||
|
useEffect(() => {
|
||||||
|
if (globalDownloadAccessData?.downloadAccess !== undefined) {
|
||||||
|
setGlobalDownloadAccess(globalDownloadAccessData.downloadAccess);
|
||||||
|
} else if (globalDownloadAccessData !== undefined && globalDownloadAccessData.downloadAccess === undefined) {
|
||||||
|
setGlobalDownloadAccess(true);
|
||||||
|
}
|
||||||
|
}, [globalDownloadAccessData]);
|
||||||
|
|
||||||
const handleGlobalAutoApproveToggle = async (newValue: boolean) => {
|
const handleGlobalAutoApproveToggle = async (newValue: boolean) => {
|
||||||
setGlobalAutoApprove(newValue);
|
setGlobalAutoApprove(newValue);
|
||||||
try {
|
try {
|
||||||
@@ -311,6 +326,43 @@ function AdminUsersPageContent() {
|
|||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
const handleGlobalDownloadAccessToggle = async (newValue: boolean) => {
|
||||||
|
setGlobalDownloadAccess(newValue);
|
||||||
|
try {
|
||||||
|
await fetchJSON('/api/admin/settings/download-access', {
|
||||||
|
method: 'PATCH',
|
||||||
|
body: JSON.stringify({ downloadAccess: newValue }),
|
||||||
|
});
|
||||||
|
toast.success(`Global download access ${newValue ? 'enabled' : 'disabled'}`);
|
||||||
|
mutateGlobalDownloadAccess();
|
||||||
|
mutate();
|
||||||
|
} catch (err) {
|
||||||
|
setGlobalDownloadAccess(!newValue);
|
||||||
|
const errorMsg = err instanceof Error ? err.message : 'Failed to update download access setting';
|
||||||
|
toast.error(errorMsg);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
const handleUserDownloadAccessToggle = async (user: User, newValue: boolean) => {
|
||||||
|
const previousUsers = data?.users || [];
|
||||||
|
const optimisticUsers = previousUsers.map((u: User) =>
|
||||||
|
u.id === user.id ? { ...u, downloadAccess: newValue } : u
|
||||||
|
);
|
||||||
|
mutate({ users: optimisticUsers }, false);
|
||||||
|
try {
|
||||||
|
await fetchJSON(`/api/admin/users/${user.id}`, {
|
||||||
|
method: 'PUT',
|
||||||
|
body: JSON.stringify({ role: user.role, downloadAccess: newValue }),
|
||||||
|
});
|
||||||
|
toast.success(`Download access ${newValue ? 'enabled' : 'disabled'} for ${user.plexUsername}`);
|
||||||
|
mutate();
|
||||||
|
} catch (err) {
|
||||||
|
mutate({ users: previousUsers }, false);
|
||||||
|
const errorMsg = err instanceof Error ? err.message : 'Failed to update user download access setting';
|
||||||
|
toast.error(errorMsg);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
const showEditDialog = (user: User) => {
|
const showEditDialog = (user: User) => {
|
||||||
setEditRole(user.role);
|
setEditRole(user.role);
|
||||||
setEditDialog({ isOpen: true, user });
|
setEditDialog({ isOpen: true, user });
|
||||||
@@ -909,6 +961,8 @@ function AdminUsersPageContent() {
|
|||||||
onToggleAutoApprove={handleGlobalAutoApproveToggle}
|
onToggleAutoApprove={handleGlobalAutoApproveToggle}
|
||||||
globalInteractiveSearch={globalInteractiveSearch}
|
globalInteractiveSearch={globalInteractiveSearch}
|
||||||
onToggleInteractiveSearch={handleGlobalInteractiveSearchToggle}
|
onToggleInteractiveSearch={handleGlobalInteractiveSearchToggle}
|
||||||
|
globalDownloadAccess={globalDownloadAccess}
|
||||||
|
onToggleDownloadAccess={handleGlobalDownloadAccessToggle}
|
||||||
/>
|
/>
|
||||||
|
|
||||||
{/* User Permissions Modal */}
|
{/* User Permissions Modal */}
|
||||||
@@ -918,12 +972,16 @@ function AdminUsersPageContent() {
|
|||||||
user={permissionsUser}
|
user={permissionsUser}
|
||||||
globalAutoApprove={globalAutoApprove}
|
globalAutoApprove={globalAutoApprove}
|
||||||
globalInteractiveSearch={globalInteractiveSearch}
|
globalInteractiveSearch={globalInteractiveSearch}
|
||||||
|
globalDownloadAccess={globalDownloadAccess}
|
||||||
onToggleAutoApprove={(user, newValue) => {
|
onToggleAutoApprove={(user, newValue) => {
|
||||||
handleUserAutoApproveToggle(user as User, newValue);
|
handleUserAutoApproveToggle(user as User, newValue);
|
||||||
}}
|
}}
|
||||||
onToggleInteractiveSearch={(user, newValue) => {
|
onToggleInteractiveSearch={(user, newValue) => {
|
||||||
handleUserInteractiveSearchToggle(user as User, newValue);
|
handleUserInteractiveSearchToggle(user as User, newValue);
|
||||||
}}
|
}}
|
||||||
|
onToggleDownloadAccess={(user, newValue) => {
|
||||||
|
handleUserDownloadAccessToggle(user as User, newValue);
|
||||||
|
}}
|
||||||
/>
|
/>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|||||||
@@ -0,0 +1,142 @@
|
|||||||
|
/**
|
||||||
|
* Component: Interactive API Documentation Page
|
||||||
|
* Documentation: documentation/backend/services/api-tokens.md
|
||||||
|
*
|
||||||
|
* Lists all API token-accessible endpoints with "Try it out" functionality.
|
||||||
|
* Users can test with a custom API token or their current browser session.
|
||||||
|
*/
|
||||||
|
|
||||||
|
'use client';
|
||||||
|
|
||||||
|
import { useState } from 'react';
|
||||||
|
import { Header } from '@/components/layout/Header';
|
||||||
|
import { ProtectedRoute } from '@/components/auth/ProtectedRoute';
|
||||||
|
import { TokenInput } from '@/components/api-docs/TokenInput';
|
||||||
|
import { EndpointCard } from '@/components/api-docs/EndpointCard';
|
||||||
|
import { API_TOKEN_ENDPOINT_DOCS } from '@/lib/constants/api-tokens';
|
||||||
|
import { useAuth } from '@/contexts/AuthContext';
|
||||||
|
import { getInstanceUrl } from '@/lib/utils/client-url';
|
||||||
|
import Link from 'next/link';
|
||||||
|
|
||||||
|
export default function ApiDocsPage() {
|
||||||
|
const { user } = useAuth();
|
||||||
|
const [token, setToken] = useState('');
|
||||||
|
const [useSession, setUseSession] = useState(false);
|
||||||
|
const isAdmin = user?.role === 'admin';
|
||||||
|
|
||||||
|
return (
|
||||||
|
<ProtectedRoute>
|
||||||
|
<div className="min-h-screen bg-gray-50 dark:bg-gray-950">
|
||||||
|
<Header />
|
||||||
|
|
||||||
|
<main className="max-w-4xl mx-auto px-4 sm:px-6 pt-8 pb-16">
|
||||||
|
{/* Page header */}
|
||||||
|
<div className="mb-8">
|
||||||
|
<div className="flex items-center gap-2 text-sm text-gray-500 dark:text-gray-400 mb-4">
|
||||||
|
<Link
|
||||||
|
href="/profile"
|
||||||
|
className="hover:text-gray-700 dark:hover:text-gray-200 transition-colors"
|
||||||
|
>
|
||||||
|
Profile
|
||||||
|
</Link>
|
||||||
|
<svg className="w-3.5 h-3.5" fill="none" stroke="currentColor" viewBox="0 0 24 24">
|
||||||
|
<path strokeLinecap="round" strokeLinejoin="round" strokeWidth={2} d="M9 5l7 7-7 7" />
|
||||||
|
</svg>
|
||||||
|
<span className="text-gray-900 dark:text-white font-medium">API Documentation</span>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<h1 className="text-3xl font-bold text-gray-900 dark:text-white tracking-tight">
|
||||||
|
API Reference
|
||||||
|
</h1>
|
||||||
|
<p className="mt-2 text-base text-gray-500 dark:text-gray-400 leading-relaxed max-w-2xl">
|
||||||
|
Interact with ReadMeABook programmatically using API tokens. These endpoints are
|
||||||
|
available for external integrations, dashboards, and automation tools.
|
||||||
|
</p>
|
||||||
|
|
||||||
|
{/* Quick links */}
|
||||||
|
<div className="flex flex-wrap gap-3 mt-4">
|
||||||
|
<Link
|
||||||
|
href="/profile"
|
||||||
|
className="inline-flex items-center gap-1.5 text-sm font-medium text-blue-600 dark:text-blue-400 hover:text-blue-700 dark:hover:text-blue-300 transition-colors"
|
||||||
|
>
|
||||||
|
<svg className="w-4 h-4" fill="none" stroke="currentColor" viewBox="0 0 24 24">
|
||||||
|
<path strokeLinecap="round" strokeLinejoin="round" strokeWidth={2} d="M15 7a2 2 0 012 2m4 0a6 6 0 01-7.743 5.743L11 17H9v2H7v2H4a1 1 0 01-1-1v-2.586a1 1 0 01.293-.707l5.964-5.964A6 6 0 1121 9z" />
|
||||||
|
</svg>
|
||||||
|
Manage your tokens
|
||||||
|
</Link>
|
||||||
|
{isAdmin && (
|
||||||
|
<>
|
||||||
|
<span className="text-gray-300 dark:text-gray-600">|</span>
|
||||||
|
<Link
|
||||||
|
href="/admin/settings"
|
||||||
|
className="inline-flex items-center gap-1.5 text-sm font-medium text-blue-600 dark:text-blue-400 hover:text-blue-700 dark:hover:text-blue-300 transition-colors"
|
||||||
|
>
|
||||||
|
<svg className="w-4 h-4" fill="none" stroke="currentColor" viewBox="0 0 24 24">
|
||||||
|
<path strokeLinecap="round" strokeLinejoin="round" strokeWidth={2} d="M10.325 4.317c.426-1.756 2.924-1.756 3.35 0a1.724 1.724 0 002.573 1.066c1.543-.94 3.31.826 2.37 2.37a1.724 1.724 0 001.066 2.573c1.756.426 1.756 2.924 0 3.35a1.724 1.724 0 00-1.066 2.573c.94 1.543-.826 3.31-2.37 2.37a1.724 1.724 0 00-2.573 1.066c-.426 1.756-2.924 1.756-3.35 0a1.724 1.724 0 00-2.573-1.066c-1.543.94-3.31-.826-2.37-2.37a1.724 1.724 0 00-1.066-2.573c-1.756-.426-1.756-2.924 0-3.35a1.724 1.724 0 001.066-2.573c-.94-1.543.826-3.31 2.37-2.37.996.608 2.296.07 2.572-1.065z" />
|
||||||
|
<path strokeLinecap="round" strokeLinejoin="round" strokeWidth={2} d="M15 12a3 3 0 11-6 0 3 3 0 016 0z" />
|
||||||
|
</svg>
|
||||||
|
Admin token management
|
||||||
|
</Link>
|
||||||
|
</>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
{/* Authentication section */}
|
||||||
|
<div className="mb-8">
|
||||||
|
<TokenInput
|
||||||
|
token={token}
|
||||||
|
onTokenChange={setToken}
|
||||||
|
useSession={useSession}
|
||||||
|
onUseSessionChange={setUseSession}
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
{/* Usage instructions card */}
|
||||||
|
<div className="mb-8 rounded-2xl border border-gray-200 dark:border-gray-700/50 bg-white dark:bg-gray-800 p-5 shadow-sm">
|
||||||
|
<h3 className="text-sm font-semibold text-gray-900 dark:text-white mb-2">
|
||||||
|
Quick Start
|
||||||
|
</h3>
|
||||||
|
<p className="text-sm text-gray-500 dark:text-gray-400 mb-3">
|
||||||
|
Include your API token in the <code className="px-1.5 py-0.5 bg-gray-100 dark:bg-gray-900 rounded text-xs font-mono">Authorization</code> header as a Bearer token:
|
||||||
|
</p>
|
||||||
|
<pre className="text-xs bg-gray-900 dark:bg-black text-gray-100 p-4 rounded-xl overflow-x-auto font-mono leading-relaxed">
|
||||||
|
{`curl -H "Authorization: Bearer rmab_your_token_here" \\
|
||||||
|
${getInstanceUrl()}/api/requests`}
|
||||||
|
</pre>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
{/* Endpoints section header */}
|
||||||
|
<div className="flex items-center gap-3 mb-5">
|
||||||
|
<h2 className="text-lg font-semibold text-gray-900 dark:text-white">
|
||||||
|
Available Endpoints
|
||||||
|
</h2>
|
||||||
|
<span className="inline-flex items-center px-2 py-0.5 rounded-md text-xs font-medium bg-gray-100 dark:bg-gray-800 text-gray-600 dark:text-gray-400">
|
||||||
|
{API_TOKEN_ENDPOINT_DOCS.length} endpoints
|
||||||
|
</span>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
{/* Endpoint cards */}
|
||||||
|
<div className="space-y-4">
|
||||||
|
{API_TOKEN_ENDPOINT_DOCS.map((endpoint) => (
|
||||||
|
<EndpointCard
|
||||||
|
key={`${endpoint.method}:${endpoint.path}`}
|
||||||
|
endpoint={endpoint}
|
||||||
|
token={token}
|
||||||
|
useSession={useSession}
|
||||||
|
/>
|
||||||
|
))}
|
||||||
|
</div>
|
||||||
|
|
||||||
|
{/* Footer note */}
|
||||||
|
<div className="mt-10 text-center">
|
||||||
|
<p className="text-xs text-gray-400 dark:text-gray-500">
|
||||||
|
API tokens are restricted to the endpoints listed above.
|
||||||
|
JWT session authentication has access to all endpoints.
|
||||||
|
</p>
|
||||||
|
</div>
|
||||||
|
</main>
|
||||||
|
</div>
|
||||||
|
</ProtectedRoute>
|
||||||
|
);
|
||||||
|
}
|
||||||
@@ -0,0 +1,56 @@
|
|||||||
|
/**
|
||||||
|
* Component: API Token Delete Route
|
||||||
|
* Documentation: documentation/backend/services/api-tokens.md
|
||||||
|
*/
|
||||||
|
|
||||||
|
import { NextRequest, NextResponse } from 'next/server';
|
||||||
|
import { requireAuth, requireAdmin, AuthenticatedRequest } from '@/lib/middleware/auth';
|
||||||
|
import { prisma } from '@/lib/db';
|
||||||
|
import { RMABLogger } from '@/lib/utils/logger';
|
||||||
|
import { checkApiTokenRevokeRateLimit } from '@/lib/utils/apiTokenRateLimit';
|
||||||
|
|
||||||
|
const logger = RMABLogger.create('API.Admin.ApiTokens');
|
||||||
|
|
||||||
|
/**
|
||||||
|
* DELETE /api/admin/api-tokens/[id]
|
||||||
|
* Revoke (delete) an API token
|
||||||
|
*/
|
||||||
|
export async function DELETE(
|
||||||
|
request: NextRequest,
|
||||||
|
{ params }: { params: Promise<{ id: string }> }
|
||||||
|
) {
|
||||||
|
return requireAuth(request, (req: AuthenticatedRequest) =>
|
||||||
|
requireAdmin(req, async () => {
|
||||||
|
try {
|
||||||
|
const rateLimit = checkApiTokenRevokeRateLimit(req.user!.id);
|
||||||
|
if (!rateLimit.allowed) {
|
||||||
|
return NextResponse.json(
|
||||||
|
{ error: 'Too many API token revoke attempts. Please try again later.' },
|
||||||
|
{
|
||||||
|
status: 429,
|
||||||
|
headers: {
|
||||||
|
'Retry-After': String(rateLimit.retryAfterSeconds),
|
||||||
|
},
|
||||||
|
}
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
const { id } = await params;
|
||||||
|
|
||||||
|
const token = await prisma.apiToken.findUnique({ where: { id } });
|
||||||
|
if (!token) {
|
||||||
|
return NextResponse.json({ error: 'Token not found' }, { status: 404 });
|
||||||
|
}
|
||||||
|
|
||||||
|
await prisma.apiToken.delete({ where: { id } });
|
||||||
|
|
||||||
|
logger.info('API token revoked', { tokenId: id, name: token.name, revokedBy: req.user!.username });
|
||||||
|
|
||||||
|
return NextResponse.json({ success: true });
|
||||||
|
} catch (error) {
|
||||||
|
logger.error('Failed to revoke API token', { error: error instanceof Error ? error.message : String(error) });
|
||||||
|
return NextResponse.json({ error: 'Failed to revoke API token' }, { status: 500 });
|
||||||
|
}
|
||||||
|
})
|
||||||
|
);
|
||||||
|
}
|
||||||
@@ -0,0 +1,190 @@
|
|||||||
|
/**
|
||||||
|
* Component: Admin API Token Management Routes
|
||||||
|
* Documentation: documentation/backend/services/api-tokens.md
|
||||||
|
*/
|
||||||
|
|
||||||
|
import { NextRequest, NextResponse } from 'next/server';
|
||||||
|
import { requireAuth, requireAdmin, AuthenticatedRequest } from '@/lib/middleware/auth';
|
||||||
|
import { prisma } from '@/lib/db';
|
||||||
|
import { RMABLogger } from '@/lib/utils/logger';
|
||||||
|
import { checkApiTokenCreateRateLimit } from '@/lib/utils/apiTokenRateLimit';
|
||||||
|
import { MAX_TOKENS_PER_USER } from '@/lib/constants/api-tokens';
|
||||||
|
import { generateApiToken } from '@/lib/utils/api-token';
|
||||||
|
import { z } from 'zod';
|
||||||
|
|
||||||
|
const logger = RMABLogger.create('API.Admin.ApiTokens');
|
||||||
|
|
||||||
|
const CreateTokenSchema = z.object({
|
||||||
|
name: z.string().min(1).max(100),
|
||||||
|
expiresAt: z.string().datetime().nullable().optional(),
|
||||||
|
userId: z.string().uuid().optional(), // Admin can specify which user the token acts as
|
||||||
|
role: z.enum(['admin', 'user']).optional(), // Accepted for compatibility, but cannot differ from target user role
|
||||||
|
});
|
||||||
|
|
||||||
|
/**
|
||||||
|
* GET /api/admin/api-tokens
|
||||||
|
* List ALL API tokens across all users
|
||||||
|
*/
|
||||||
|
export async function GET(request: NextRequest) {
|
||||||
|
return requireAuth(request, (req: AuthenticatedRequest) =>
|
||||||
|
requireAdmin(req, async () => {
|
||||||
|
try {
|
||||||
|
const tokens = await prisma.apiToken.findMany({
|
||||||
|
include: {
|
||||||
|
createdBy: {
|
||||||
|
select: { id: true, plexUsername: true },
|
||||||
|
},
|
||||||
|
tokenUser: {
|
||||||
|
select: { id: true, plexUsername: true, role: true },
|
||||||
|
},
|
||||||
|
},
|
||||||
|
orderBy: { createdAt: 'desc' },
|
||||||
|
});
|
||||||
|
|
||||||
|
const sanitized = tokens.map((t) => ({
|
||||||
|
id: t.id,
|
||||||
|
name: t.name,
|
||||||
|
tokenPrefix: t.tokenPrefix,
|
||||||
|
role: t.role,
|
||||||
|
createdBy: t.createdBy.plexUsername,
|
||||||
|
createdById: t.createdBy.id,
|
||||||
|
tokenUser: t.tokenUser.plexUsername,
|
||||||
|
tokenUserId: t.tokenUser.id,
|
||||||
|
lastUsedAt: t.lastUsedAt,
|
||||||
|
expiresAt: t.expiresAt,
|
||||||
|
createdAt: t.createdAt,
|
||||||
|
}));
|
||||||
|
|
||||||
|
return NextResponse.json({ tokens: sanitized });
|
||||||
|
} catch (error) {
|
||||||
|
logger.error('Failed to list API tokens', { error: error instanceof Error ? error.message : String(error) });
|
||||||
|
return NextResponse.json({ error: 'Failed to list API tokens' }, { status: 500 });
|
||||||
|
}
|
||||||
|
})
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* POST /api/admin/api-tokens
|
||||||
|
* Create a new API token. Admin can optionally specify userId.
|
||||||
|
* Token role is always derived from the target user's current role.
|
||||||
|
* Returns the full token ONCE.
|
||||||
|
*/
|
||||||
|
export async function POST(request: NextRequest) {
|
||||||
|
return requireAuth(request, (req: AuthenticatedRequest) =>
|
||||||
|
requireAdmin(req, async () => {
|
||||||
|
try {
|
||||||
|
const rateLimit = checkApiTokenCreateRateLimit(req.user!.id);
|
||||||
|
if (!rateLimit.allowed) {
|
||||||
|
return NextResponse.json(
|
||||||
|
{ error: 'Too many API token create attempts. Please try again later.' },
|
||||||
|
{
|
||||||
|
status: 429,
|
||||||
|
headers: {
|
||||||
|
'Retry-After': String(rateLimit.retryAfterSeconds),
|
||||||
|
},
|
||||||
|
}
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
const body = await req.json();
|
||||||
|
const { name, expiresAt, userId, role } = CreateTokenSchema.parse(body);
|
||||||
|
|
||||||
|
// Determine target user (defaults to the admin themselves)
|
||||||
|
const targetUserId = userId || req.user!.id;
|
||||||
|
|
||||||
|
// Verify the target user exists
|
||||||
|
const targetUser = await prisma.user.findUnique({
|
||||||
|
where: { id: targetUserId },
|
||||||
|
select: { id: true, role: true, plexUsername: true },
|
||||||
|
});
|
||||||
|
|
||||||
|
if (!targetUser) {
|
||||||
|
return NextResponse.json({ error: 'Target user not found' }, { status: 404 });
|
||||||
|
}
|
||||||
|
|
||||||
|
// Enforce per-user token cap (count only active, non-expired tokens)
|
||||||
|
const activeTokenCount = await prisma.apiToken.count({
|
||||||
|
where: {
|
||||||
|
userId: targetUserId,
|
||||||
|
OR: [
|
||||||
|
{ expiresAt: null },
|
||||||
|
{ expiresAt: { gt: new Date() } },
|
||||||
|
],
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
if (activeTokenCount >= MAX_TOKENS_PER_USER) {
|
||||||
|
return NextResponse.json(
|
||||||
|
{ error: `Token limit reached. Users may have at most ${MAX_TOKENS_PER_USER} active API tokens.` },
|
||||||
|
{ status: 403 }
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Security guard: token role must always match the target user's persisted role.
|
||||||
|
// This avoids role/identity mismatch (for example: acting as user A with admin role).
|
||||||
|
if (role && role !== targetUser.role) {
|
||||||
|
logger.warn('Admin attempted token role override that differs from target user role', {
|
||||||
|
requestedRole: role,
|
||||||
|
userActualRole: targetUser.role,
|
||||||
|
targetUser: targetUser.plexUsername,
|
||||||
|
createdBy: req.user!.username,
|
||||||
|
});
|
||||||
|
|
||||||
|
return NextResponse.json(
|
||||||
|
{
|
||||||
|
error: `Token role must match target user's role (${targetUser.role}).`,
|
||||||
|
},
|
||||||
|
{ status: 400 }
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
const tokenRole = targetUser.role;
|
||||||
|
|
||||||
|
// Generate the token
|
||||||
|
const { fullToken, tokenHash, tokenPrefix } = generateApiToken();
|
||||||
|
|
||||||
|
const apiToken = await prisma.apiToken.create({
|
||||||
|
data: {
|
||||||
|
name,
|
||||||
|
tokenHash,
|
||||||
|
tokenPrefix,
|
||||||
|
role: tokenRole,
|
||||||
|
createdById: req.user!.id,
|
||||||
|
userId: targetUserId,
|
||||||
|
expiresAt: expiresAt ? new Date(expiresAt) : null,
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
logger.info('Admin API token created', {
|
||||||
|
tokenId: apiToken.id,
|
||||||
|
name,
|
||||||
|
createdBy: req.user!.username,
|
||||||
|
targetUser: targetUser.plexUsername,
|
||||||
|
role: tokenRole,
|
||||||
|
});
|
||||||
|
|
||||||
|
return NextResponse.json({
|
||||||
|
token: {
|
||||||
|
id: apiToken.id,
|
||||||
|
name: apiToken.name,
|
||||||
|
tokenPrefix: apiToken.tokenPrefix,
|
||||||
|
role: apiToken.role,
|
||||||
|
expiresAt: apiToken.expiresAt,
|
||||||
|
createdAt: apiToken.createdAt,
|
||||||
|
},
|
||||||
|
// Full token is returned ONLY on creation
|
||||||
|
fullToken,
|
||||||
|
}, { status: 201 });
|
||||||
|
} catch (error) {
|
||||||
|
logger.error('Failed to create API token', { error: error instanceof Error ? error.message : String(error) });
|
||||||
|
|
||||||
|
if (error instanceof z.ZodError) {
|
||||||
|
return NextResponse.json({ error: 'Validation error', details: error.errors }, { status: 400 });
|
||||||
|
}
|
||||||
|
|
||||||
|
return NextResponse.json({ error: 'Failed to create API token' }, { status: 500 });
|
||||||
|
}
|
||||||
|
})
|
||||||
|
);
|
||||||
|
}
|
||||||
@@ -0,0 +1,208 @@
|
|||||||
|
/**
|
||||||
|
* Component: Admin Filesystem Browse API
|
||||||
|
* Documentation: documentation/features/manual-import.md
|
||||||
|
*
|
||||||
|
* Lets admins browse server directories for manual audiobook import.
|
||||||
|
* Restricted to download_dir and media_dir roots only.
|
||||||
|
*/
|
||||||
|
|
||||||
|
import { NextRequest, NextResponse } from 'next/server';
|
||||||
|
import { requireAuth, requireAdmin, AuthenticatedRequest } from '@/lib/middleware/auth';
|
||||||
|
import { prisma } from '@/lib/db';
|
||||||
|
import { RMABLogger } from '@/lib/utils/logger';
|
||||||
|
import { AUDIO_EXTENSIONS } from '@/lib/constants/audio-formats';
|
||||||
|
|
||||||
|
const logger = RMABLogger.create('API.Admin.Filesystem.Browse');
|
||||||
|
|
||||||
|
interface DirectoryEntry {
|
||||||
|
name: string;
|
||||||
|
type: 'directory';
|
||||||
|
audioFileCount: number;
|
||||||
|
subfolderCount: number;
|
||||||
|
totalSize: number;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Scan immediate children of a directory to gather audio file and subfolder stats.
|
||||||
|
*/
|
||||||
|
async function getDirectoryStats(
|
||||||
|
dirPath: string
|
||||||
|
): Promise<{ audioFileCount: number; subfolderCount: number; totalSize: number }> {
|
||||||
|
const fs = await import('fs/promises');
|
||||||
|
const pathModule = await import('path');
|
||||||
|
|
||||||
|
let audioFileCount = 0;
|
||||||
|
let subfolderCount = 0;
|
||||||
|
let totalSize = 0;
|
||||||
|
|
||||||
|
try {
|
||||||
|
const children = await fs.readdir(dirPath, { withFileTypes: true });
|
||||||
|
for (const child of children) {
|
||||||
|
if (child.isDirectory()) {
|
||||||
|
subfolderCount++;
|
||||||
|
} else if (child.isFile()) {
|
||||||
|
const ext = pathModule.extname(child.name).toLowerCase();
|
||||||
|
if ((AUDIO_EXTENSIONS as readonly string[]).includes(ext)) {
|
||||||
|
audioFileCount++;
|
||||||
|
try {
|
||||||
|
const stat = await fs.stat(pathModule.join(dirPath, child.name));
|
||||||
|
totalSize += stat.size;
|
||||||
|
} catch {
|
||||||
|
/* skip unreadable files */
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} catch {
|
||||||
|
/* directory not readable */
|
||||||
|
}
|
||||||
|
|
||||||
|
return { audioFileCount, subfolderCount, totalSize };
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Load allowed root directories from Configuration table.
|
||||||
|
*/
|
||||||
|
const BOOKDROP_PATH = '/bookdrop';
|
||||||
|
|
||||||
|
async function getAllowedRoots(): Promise<{ downloadDir: string | null; mediaDir: string | null; bookdropExists: boolean }> {
|
||||||
|
const downloadDirConfig = await prisma.configuration.findUnique({
|
||||||
|
where: { key: 'download_dir' },
|
||||||
|
});
|
||||||
|
const mediaDirConfig = await prisma.configuration.findUnique({
|
||||||
|
where: { key: 'media_dir' },
|
||||||
|
});
|
||||||
|
|
||||||
|
let bookdropExists = false;
|
||||||
|
try {
|
||||||
|
const fs = await import('fs/promises');
|
||||||
|
const stat = await fs.stat(BOOKDROP_PATH);
|
||||||
|
bookdropExists = stat.isDirectory();
|
||||||
|
} catch {
|
||||||
|
/* not mounted */
|
||||||
|
}
|
||||||
|
|
||||||
|
return {
|
||||||
|
downloadDir: downloadDirConfig?.value || null,
|
||||||
|
mediaDir: mediaDirConfig?.value || null,
|
||||||
|
bookdropExists,
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Check if a normalized path is within one of the allowed roots.
|
||||||
|
*/
|
||||||
|
function isPathAllowed(normalizedPath: string, roots: string[]): boolean {
|
||||||
|
return roots.some(
|
||||||
|
(root) => normalizedPath === root || normalizedPath.startsWith(root + '/')
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
export async function GET(request: NextRequest) {
|
||||||
|
return requireAuth(request, async (req: AuthenticatedRequest) => {
|
||||||
|
return requireAdmin(req, async () => {
|
||||||
|
try {
|
||||||
|
const pathModule = await import('path');
|
||||||
|
const fs = await import('fs/promises');
|
||||||
|
|
||||||
|
const { downloadDir, mediaDir, bookdropExists } = await getAllowedRoots();
|
||||||
|
const requestedPath = request.nextUrl.searchParams.get('path');
|
||||||
|
|
||||||
|
// No path param: return root directories
|
||||||
|
if (!requestedPath) {
|
||||||
|
const roots: Array<{ name: string; path: string; icon: string }> = [];
|
||||||
|
if (downloadDir) {
|
||||||
|
roots.push({ name: 'Downloads', path: downloadDir, icon: 'download' });
|
||||||
|
}
|
||||||
|
if (mediaDir) {
|
||||||
|
roots.push({ name: 'Media Library', path: mediaDir, icon: 'library' });
|
||||||
|
}
|
||||||
|
if (bookdropExists) {
|
||||||
|
roots.push({ name: 'Book Drop', path: BOOKDROP_PATH, icon: 'bookdrop' });
|
||||||
|
}
|
||||||
|
|
||||||
|
if (roots.length === 0) {
|
||||||
|
return NextResponse.json(
|
||||||
|
{ error: 'No browsable directories available' },
|
||||||
|
{ status: 400 }
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
return NextResponse.json({ roots });
|
||||||
|
}
|
||||||
|
|
||||||
|
// Path param provided: browse that directory
|
||||||
|
// Normalize to forward slashes and resolve
|
||||||
|
const normalizedPath = pathModule.resolve(requestedPath).replace(/\\/g, '/');
|
||||||
|
|
||||||
|
// Build list of allowed roots (normalized)
|
||||||
|
const allowedRoots: string[] = [];
|
||||||
|
if (downloadDir) allowedRoots.push(pathModule.resolve(downloadDir).replace(/\\/g, '/'));
|
||||||
|
if (mediaDir) allowedRoots.push(pathModule.resolve(mediaDir).replace(/\\/g, '/'));
|
||||||
|
if (bookdropExists) allowedRoots.push(pathModule.resolve(BOOKDROP_PATH).replace(/\\/g, '/'));
|
||||||
|
|
||||||
|
if (!isPathAllowed(normalizedPath, allowedRoots)) {
|
||||||
|
logger.warn(`Access denied: ${normalizedPath} is outside allowed directories`);
|
||||||
|
return NextResponse.json(
|
||||||
|
{ error: 'Access denied: path outside allowed directories' },
|
||||||
|
{ status: 403 }
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Read directory entries
|
||||||
|
const dirEntries = await fs.readdir(normalizedPath, { withFileTypes: true });
|
||||||
|
|
||||||
|
// Gather stats for each subdirectory (parallel for performance)
|
||||||
|
const directoryEntries = dirEntries.filter((e) => e.isDirectory());
|
||||||
|
const statsPromises = directoryEntries.map(async (entry): Promise<DirectoryEntry> => {
|
||||||
|
const fullPath = pathModule.join(normalizedPath, entry.name);
|
||||||
|
const stats = await getDirectoryStats(fullPath);
|
||||||
|
return {
|
||||||
|
name: entry.name,
|
||||||
|
type: 'directory',
|
||||||
|
...stats,
|
||||||
|
};
|
||||||
|
});
|
||||||
|
|
||||||
|
const entries = await Promise.all(statsPromises);
|
||||||
|
entries.sort((a, b) => a.name.localeCompare(b.name));
|
||||||
|
|
||||||
|
// Gather audio files in the current directory
|
||||||
|
const audioFiles: Array<{ name: string; size: number }> = [];
|
||||||
|
for (const entry of dirEntries) {
|
||||||
|
if (entry.isFile()) {
|
||||||
|
const ext = pathModule.extname(entry.name).toLowerCase();
|
||||||
|
if ((AUDIO_EXTENSIONS as readonly string[]).includes(ext)) {
|
||||||
|
try {
|
||||||
|
const stat = await fs.stat(pathModule.join(normalizedPath, entry.name));
|
||||||
|
audioFiles.push({ name: entry.name, size: stat.size });
|
||||||
|
} catch {
|
||||||
|
audioFiles.push({ name: entry.name, size: 0 });
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
audioFiles.sort((a, b) => a.name.localeCompare(b.name));
|
||||||
|
|
||||||
|
return NextResponse.json({ path: normalizedPath, entries, audioFiles });
|
||||||
|
} catch (error) {
|
||||||
|
const code = (error as NodeJS.ErrnoException).code;
|
||||||
|
|
||||||
|
if (code === 'ENOENT') {
|
||||||
|
return NextResponse.json({ error: 'Directory not found' }, { status: 404 });
|
||||||
|
}
|
||||||
|
if (code === 'EACCES' || code === 'EPERM') {
|
||||||
|
return NextResponse.json({ error: 'Permission denied' }, { status: 403 });
|
||||||
|
}
|
||||||
|
|
||||||
|
logger.error('Failed to browse directory', {
|
||||||
|
error: error instanceof Error ? error.message : String(error),
|
||||||
|
});
|
||||||
|
return NextResponse.json(
|
||||||
|
{ error: 'Failed to browse directory' },
|
||||||
|
{ status: 500 }
|
||||||
|
);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
});
|
||||||
|
}
|
||||||
@@ -0,0 +1,340 @@
|
|||||||
|
/**
|
||||||
|
* Component: Admin Manual Import API
|
||||||
|
* Documentation: documentation/features/manual-import.md
|
||||||
|
*
|
||||||
|
* Triggers the organize_files pipeline for a manually-selected folder.
|
||||||
|
* Creates or recycles a request, then queues the organize job.
|
||||||
|
*/
|
||||||
|
|
||||||
|
import { NextRequest, NextResponse } from 'next/server';
|
||||||
|
import { requireAuth, requireAdmin, AuthenticatedRequest } from '@/lib/middleware/auth';
|
||||||
|
import { prisma } from '@/lib/db';
|
||||||
|
import { getJobQueueService } from '@/lib/services/job-queue.service';
|
||||||
|
import { RMABLogger } from '@/lib/utils/logger';
|
||||||
|
import { AUDIO_EXTENSIONS } from '@/lib/constants/audio-formats';
|
||||||
|
import { getAudibleService } from '@/lib/integrations/audible.service';
|
||||||
|
|
||||||
|
const logger = RMABLogger.create('API.Admin.ManualImport');
|
||||||
|
|
||||||
|
/** Statuses that indicate the request is actively being worked on. */
|
||||||
|
const ACTIVE_STATUSES = ['searching', 'downloading', 'processing', 'awaiting_import'];
|
||||||
|
|
||||||
|
/** Statuses that can be recycled for a new manual import. */
|
||||||
|
const RECYCLABLE_STATUSES = ['failed', 'warn', 'cancelled', 'denied', 'pending', 'awaiting_search', 'awaiting_approval'];
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Check if a directory contains at least one audio file (immediate children only).
|
||||||
|
*/
|
||||||
|
async function hasAudioFiles(dirPath: string): Promise<{ found: boolean; count: number }> {
|
||||||
|
const fs = await import('fs/promises');
|
||||||
|
const pathModule = await import('path');
|
||||||
|
|
||||||
|
let count = 0;
|
||||||
|
try {
|
||||||
|
const children = await fs.readdir(dirPath, { withFileTypes: true });
|
||||||
|
for (const child of children) {
|
||||||
|
if (child.isFile()) {
|
||||||
|
const ext = pathModule.extname(child.name).toLowerCase();
|
||||||
|
if ((AUDIO_EXTENSIONS as readonly string[]).includes(ext)) {
|
||||||
|
count++;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} catch {
|
||||||
|
/* directory not readable */
|
||||||
|
}
|
||||||
|
|
||||||
|
return { found: count > 0, count };
|
||||||
|
}
|
||||||
|
|
||||||
|
export async function POST(request: NextRequest) {
|
||||||
|
return requireAuth(request, async (req: AuthenticatedRequest) => {
|
||||||
|
return requireAdmin(req, async () => {
|
||||||
|
try {
|
||||||
|
const pathModule = await import('path');
|
||||||
|
const fs = await import('fs/promises');
|
||||||
|
|
||||||
|
const body = await request.json();
|
||||||
|
const { folderPath, asin, cleanupSource } = body;
|
||||||
|
let { audiobookId } = body;
|
||||||
|
|
||||||
|
// Validate required fields
|
||||||
|
if ((!audiobookId && !asin) || !folderPath) {
|
||||||
|
return NextResponse.json(
|
||||||
|
{ error: 'folderPath and either audiobookId or asin are required' },
|
||||||
|
{ status: 400 }
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Load allowed roots
|
||||||
|
const BOOKDROP_PATH = '/bookdrop';
|
||||||
|
const downloadDirConfig = await prisma.configuration.findUnique({
|
||||||
|
where: { key: 'download_dir' },
|
||||||
|
});
|
||||||
|
const mediaDirConfig = await prisma.configuration.findUnique({
|
||||||
|
where: { key: 'media_dir' },
|
||||||
|
});
|
||||||
|
|
||||||
|
const allowedRoots: string[] = [];
|
||||||
|
if (downloadDirConfig?.value) {
|
||||||
|
allowedRoots.push(pathModule.resolve(downloadDirConfig.value).replace(/\\/g, '/'));
|
||||||
|
}
|
||||||
|
if (mediaDirConfig?.value) {
|
||||||
|
allowedRoots.push(pathModule.resolve(mediaDirConfig.value).replace(/\\/g, '/'));
|
||||||
|
}
|
||||||
|
try {
|
||||||
|
const bookdropStat = await fs.stat(BOOKDROP_PATH);
|
||||||
|
if (bookdropStat.isDirectory()) {
|
||||||
|
allowedRoots.push(pathModule.resolve(BOOKDROP_PATH).replace(/\\/g, '/'));
|
||||||
|
}
|
||||||
|
} catch {
|
||||||
|
/* not mounted */
|
||||||
|
}
|
||||||
|
|
||||||
|
// Normalize and validate path
|
||||||
|
const normalizedPath = pathModule.resolve(folderPath).replace(/\\/g, '/');
|
||||||
|
const isAllowed = allowedRoots.some(
|
||||||
|
(root) => normalizedPath === root || normalizedPath.startsWith(root + '/')
|
||||||
|
);
|
||||||
|
|
||||||
|
if (!isAllowed) {
|
||||||
|
return NextResponse.json(
|
||||||
|
{ error: 'Access denied: path outside allowed directories' },
|
||||||
|
{ status: 403 }
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Verify folder exists and is a directory
|
||||||
|
try {
|
||||||
|
const stat = await fs.stat(normalizedPath);
|
||||||
|
if (!stat.isDirectory()) {
|
||||||
|
return NextResponse.json(
|
||||||
|
{ error: 'Path is not a directory' },
|
||||||
|
{ status: 400 }
|
||||||
|
);
|
||||||
|
}
|
||||||
|
} catch {
|
||||||
|
return NextResponse.json(
|
||||||
|
{ error: 'Directory not found' },
|
||||||
|
{ status: 404 }
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Verify folder contains audio files
|
||||||
|
const audioCheck = await hasAudioFiles(normalizedPath);
|
||||||
|
if (!audioCheck.found) {
|
||||||
|
return NextResponse.json(
|
||||||
|
{ error: 'No audio files found in the selected directory' },
|
||||||
|
{ status: 400 }
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Resolve audiobook by ASIN if audiobookId not provided
|
||||||
|
if (!audiobookId && asin) {
|
||||||
|
const byAsin = await prisma.audiobook.findFirst({
|
||||||
|
where: { audibleAsin: asin },
|
||||||
|
});
|
||||||
|
if (byAsin) {
|
||||||
|
audiobookId = byAsin.id;
|
||||||
|
} else {
|
||||||
|
// Create audiobook record from Audible cache if available
|
||||||
|
const cached = await prisma.audibleCache.findUnique({
|
||||||
|
where: { asin },
|
||||||
|
});
|
||||||
|
if (cached) {
|
||||||
|
const newBook = await prisma.audiobook.create({
|
||||||
|
data: {
|
||||||
|
audibleAsin: asin,
|
||||||
|
title: cached.title,
|
||||||
|
author: cached.author,
|
||||||
|
coverArtUrl: cached.coverArtUrl,
|
||||||
|
narrator: cached.narrator,
|
||||||
|
status: 'pending',
|
||||||
|
},
|
||||||
|
});
|
||||||
|
audiobookId = newBook.id;
|
||||||
|
logger.info(`Created audiobook record from cache for ASIN ${asin}: ${newBook.id}`);
|
||||||
|
} else {
|
||||||
|
// Not in DB — fetch live from Audnexus and create a record
|
||||||
|
try {
|
||||||
|
const audibleService = getAudibleService();
|
||||||
|
const liveData = await audibleService.getAudiobookDetails(asin);
|
||||||
|
if (liveData) {
|
||||||
|
const newBook = await prisma.audiobook.create({
|
||||||
|
data: {
|
||||||
|
audibleAsin: asin,
|
||||||
|
title: liveData.title,
|
||||||
|
author: liveData.author,
|
||||||
|
coverArtUrl: liveData.coverArtUrl,
|
||||||
|
narrator: liveData.narrator,
|
||||||
|
series: liveData.series,
|
||||||
|
seriesPart: liveData.seriesPart,
|
||||||
|
seriesAsin: liveData.seriesAsin,
|
||||||
|
year: liveData.releaseDate
|
||||||
|
? new Date(liveData.releaseDate).getFullYear() || undefined
|
||||||
|
: undefined,
|
||||||
|
status: 'pending',
|
||||||
|
},
|
||||||
|
});
|
||||||
|
audiobookId = newBook.id;
|
||||||
|
logger.info(`Created audiobook record from Audnexus for ASIN ${asin}: ${newBook.id}`);
|
||||||
|
} else {
|
||||||
|
return NextResponse.json(
|
||||||
|
{ error: 'Audiobook not found for the given ASIN' },
|
||||||
|
{ status: 404 }
|
||||||
|
);
|
||||||
|
}
|
||||||
|
} catch (audnexusError) {
|
||||||
|
logger.error(`Failed to fetch ASIN ${asin} from Audnexus: ${audnexusError instanceof Error ? audnexusError.message : String(audnexusError)}`);
|
||||||
|
return NextResponse.json(
|
||||||
|
{ error: 'Audiobook not found for the given ASIN' },
|
||||||
|
{ status: 404 }
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Verify audiobook exists
|
||||||
|
const audiobook = await prisma.audiobook.findUnique({
|
||||||
|
where: { id: audiobookId },
|
||||||
|
});
|
||||||
|
|
||||||
|
if (!audiobook) {
|
||||||
|
return NextResponse.json(
|
||||||
|
{ error: 'Audiobook not found' },
|
||||||
|
{ status: 404 }
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Enrich missing series/year data from Audnexus (mirrors request-creator.service.ts)
|
||||||
|
if (audiobook.audibleAsin && (!audiobook.series || !audiobook.year)) {
|
||||||
|
try {
|
||||||
|
const audibleService = getAudibleService();
|
||||||
|
const audnexusData = await audibleService.getAudiobookDetails(audiobook.audibleAsin);
|
||||||
|
|
||||||
|
if (audnexusData) {
|
||||||
|
const updates: Record<string, any> = {};
|
||||||
|
|
||||||
|
if (!audiobook.series && audnexusData.series) {
|
||||||
|
updates.series = audnexusData.series;
|
||||||
|
}
|
||||||
|
if (!audiobook.seriesPart && audnexusData.seriesPart) {
|
||||||
|
updates.seriesPart = audnexusData.seriesPart;
|
||||||
|
}
|
||||||
|
if (!audiobook.seriesAsin && audnexusData.seriesAsin) {
|
||||||
|
updates.seriesAsin = audnexusData.seriesAsin;
|
||||||
|
}
|
||||||
|
if (!audiobook.year && audnexusData.releaseDate) {
|
||||||
|
const releaseYear = new Date(audnexusData.releaseDate).getFullYear();
|
||||||
|
if (!isNaN(releaseYear)) {
|
||||||
|
updates.year = releaseYear;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (!audiobook.narrator && audnexusData.narrator) {
|
||||||
|
updates.narrator = audnexusData.narrator;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (Object.keys(updates).length > 0) {
|
||||||
|
await prisma.audiobook.update({
|
||||||
|
where: { id: audiobook.id },
|
||||||
|
data: updates,
|
||||||
|
});
|
||||||
|
logger.info(`Enriched audiobook metadata from Audnexus for ASIN ${audiobook.audibleAsin}`, updates);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} catch (error) {
|
||||||
|
// Non-fatal: series enrichment failure should never block the import
|
||||||
|
logger.warn(`Failed to enrich metadata from Audnexus for ASIN ${audiobook.audibleAsin}: ${error instanceof Error ? error.message : String(error)}`);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Check for existing requests
|
||||||
|
const existingRequest = await prisma.request.findFirst({
|
||||||
|
where: {
|
||||||
|
audiobookId,
|
||||||
|
type: 'audiobook',
|
||||||
|
deletedAt: null,
|
||||||
|
},
|
||||||
|
orderBy: { createdAt: 'desc' },
|
||||||
|
});
|
||||||
|
|
||||||
|
let requestId: string;
|
||||||
|
|
||||||
|
if (existingRequest) {
|
||||||
|
// Check if already in an active state
|
||||||
|
if (ACTIVE_STATUSES.includes(existingRequest.status)) {
|
||||||
|
return NextResponse.json(
|
||||||
|
{ error: 'This audiobook is already being processed' },
|
||||||
|
{ status: 409 }
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Recycle the existing request
|
||||||
|
if (RECYCLABLE_STATUSES.includes(existingRequest.status) ||
|
||||||
|
existingRequest.status === 'downloaded' ||
|
||||||
|
existingRequest.status === 'available') {
|
||||||
|
await prisma.request.update({
|
||||||
|
where: { id: existingRequest.id },
|
||||||
|
data: {
|
||||||
|
status: 'processing',
|
||||||
|
progress: 100,
|
||||||
|
errorMessage: null,
|
||||||
|
importAttempts: 0,
|
||||||
|
updatedAt: new Date(),
|
||||||
|
},
|
||||||
|
});
|
||||||
|
requestId = existingRequest.id;
|
||||||
|
logger.info(`Recycled existing request ${requestId} for manual import`);
|
||||||
|
} else {
|
||||||
|
// Unknown status - create new
|
||||||
|
const newRequest = await prisma.request.create({
|
||||||
|
data: {
|
||||||
|
userId: req.user!.id,
|
||||||
|
audiobookId,
|
||||||
|
type: 'audiobook',
|
||||||
|
status: 'processing',
|
||||||
|
progress: 100,
|
||||||
|
},
|
||||||
|
});
|
||||||
|
requestId = newRequest.id;
|
||||||
|
logger.info(`Created new request ${requestId} (existing had status: ${existingRequest.status})`);
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
// No existing request - create one
|
||||||
|
const newRequest = await prisma.request.create({
|
||||||
|
data: {
|
||||||
|
userId: req.user!.id,
|
||||||
|
audiobookId,
|
||||||
|
type: 'audiobook',
|
||||||
|
status: 'processing',
|
||||||
|
progress: 100,
|
||||||
|
},
|
||||||
|
});
|
||||||
|
requestId = newRequest.id;
|
||||||
|
logger.info(`Created new request ${requestId} for manual import`);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Queue organize_files job
|
||||||
|
const jobQueue = getJobQueueService();
|
||||||
|
await jobQueue.addOrganizeJob(requestId, audiobookId, normalizedPath, undefined, cleanupSource === true);
|
||||||
|
|
||||||
|
logger.info(`Manual import queued: request=${requestId}, path=${normalizedPath}, audioFiles=${audioCheck.count}`);
|
||||||
|
|
||||||
|
return NextResponse.json({
|
||||||
|
success: true,
|
||||||
|
requestId,
|
||||||
|
message: `Import started for ${audiobook.title}`,
|
||||||
|
});
|
||||||
|
} catch (error) {
|
||||||
|
logger.error('Manual import failed', {
|
||||||
|
error: error instanceof Error ? error.message : String(error),
|
||||||
|
});
|
||||||
|
return NextResponse.json(
|
||||||
|
{ error: error instanceof Error ? error.message : 'Manual import failed' },
|
||||||
|
{ status: 500 }
|
||||||
|
);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
});
|
||||||
|
}
|
||||||
@@ -14,6 +14,7 @@ const logger = RMABLogger.create('API.Admin.Requests.Approve');
|
|||||||
|
|
||||||
const ApprovalActionSchema = z.object({
|
const ApprovalActionSchema = z.object({
|
||||||
action: z.enum(['approve', 'deny']),
|
action: z.enum(['approve', 'deny']),
|
||||||
|
selectedTorrent: z.any().optional(),
|
||||||
});
|
});
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -37,8 +38,8 @@ export async function POST(
|
|||||||
const { id } = await params;
|
const { id } = await params;
|
||||||
const body = await request.json();
|
const body = await request.json();
|
||||||
|
|
||||||
// Validate action
|
// Validate action and optional admin-selected torrent
|
||||||
const { action } = ApprovalActionSchema.parse(body);
|
const { action, selectedTorrent: adminSelectedTorrent } = ApprovalActionSchema.parse(body);
|
||||||
|
|
||||||
// Fetch the request
|
// Fetch the request
|
||||||
const existingRequest = await prisma.request.findUnique({
|
const existingRequest = await prisma.request.findUnique({
|
||||||
@@ -78,12 +79,15 @@ export async function POST(
|
|||||||
const jobQueue = getJobQueueService();
|
const jobQueue = getJobQueueService();
|
||||||
const isEbookRequest = existingRequest.type === 'ebook';
|
const isEbookRequest = existingRequest.type === 'ebook';
|
||||||
|
|
||||||
// Check if request has a pre-selected torrent (from interactive search)
|
// Use admin-provided torrent (from admin interactive search) or fall back to user's pre-selected torrent
|
||||||
if (existingRequest.selectedTorrent) {
|
const effectiveTorrent = adminSelectedTorrent || existingRequest.selectedTorrent;
|
||||||
const selectedTorrent = existingRequest.selectedTorrent as any;
|
|
||||||
|
|
||||||
// User pre-selected a specific torrent - download that torrent directly
|
if (effectiveTorrent) {
|
||||||
logger.info(`Request ${id} has pre-selected torrent, starting download`, {
|
const selectedTorrent = effectiveTorrent as any;
|
||||||
|
const torrentSource = adminSelectedTorrent ? 'admin' : 'user';
|
||||||
|
|
||||||
|
// Download the selected torrent directly
|
||||||
|
logger.info(`Request ${id} has ${torrentSource}-selected torrent, starting download`, {
|
||||||
requestId: id,
|
requestId: id,
|
||||||
userId: existingRequest.userId,
|
userId: existingRequest.userId,
|
||||||
adminId: req.user.sub,
|
adminId: req.user.sub,
|
||||||
@@ -167,17 +171,20 @@ export async function POST(
|
|||||||
logger.error('Failed to queue notification', { error: error instanceof Error ? error.message : String(error) });
|
logger.error('Failed to queue notification', { error: error instanceof Error ? error.message : String(error) });
|
||||||
});
|
});
|
||||||
|
|
||||||
logger.info(`Request ${id} approved by admin ${req.user.sub}, downloading pre-selected torrent`, {
|
logger.info(`Request ${id} approved by admin ${req.user.sub}, downloading ${torrentSource}-selected torrent`, {
|
||||||
requestId: id,
|
requestId: id,
|
||||||
userId: updatedRequest.userId,
|
userId: updatedRequest.userId,
|
||||||
audiobookTitle: existingRequest.audiobook.title,
|
audiobookTitle: existingRequest.audiobook.title,
|
||||||
adminId: req.user.sub,
|
adminId: req.user.sub,
|
||||||
type: existingRequest.type,
|
type: existingRequest.type,
|
||||||
|
torrentSource,
|
||||||
});
|
});
|
||||||
|
|
||||||
return NextResponse.json({
|
return NextResponse.json({
|
||||||
success: true,
|
success: true,
|
||||||
message: 'Request approved and download started with pre-selected torrent',
|
message: adminSelectedTorrent
|
||||||
|
? 'Request approved and download started with admin-selected torrent'
|
||||||
|
: 'Request approved and download started with pre-selected torrent',
|
||||||
request: updatedRequest,
|
request: updatedRequest,
|
||||||
});
|
});
|
||||||
} else {
|
} else {
|
||||||
|
|||||||
@@ -0,0 +1,271 @@
|
|||||||
|
/**
|
||||||
|
* Component: Admin Retry Download API
|
||||||
|
* Documentation: documentation/admin-dashboard.md
|
||||||
|
*
|
||||||
|
* Retries a failed download by either resuming monitoring of a still-alive
|
||||||
|
* download in the client, or re-adding the download using metadata from the
|
||||||
|
* most recent selected DownloadHistory record.
|
||||||
|
*/
|
||||||
|
|
||||||
|
import { NextRequest, NextResponse } from 'next/server';
|
||||||
|
import { requireAuth, requireAdmin, AuthenticatedRequest } from '@/lib/middleware/auth';
|
||||||
|
import { prisma } from '@/lib/db';
|
||||||
|
import { getJobQueueService } from '@/lib/services/job-queue.service';
|
||||||
|
import { getConfigService } from '@/lib/services/config.service';
|
||||||
|
import { getDownloadClientManager } from '@/lib/services/download-client-manager.service';
|
||||||
|
import { CLIENT_PROTOCOL_MAP, DownloadClientType } from '@/lib/interfaces/download-client.interface';
|
||||||
|
import { TorrentResult } from '@/lib/utils/ranking-algorithm';
|
||||||
|
import { RMABLogger } from '@/lib/utils/logger';
|
||||||
|
|
||||||
|
const logger = RMABLogger.create('API.Admin.Requests.RetryDownload');
|
||||||
|
|
||||||
|
/** Download statuses considered "alive" — monitoring can be resumed */
|
||||||
|
const ALIVE_STATUSES = new Set([
|
||||||
|
'downloading',
|
||||||
|
'queued',
|
||||||
|
'paused',
|
||||||
|
'checking',
|
||||||
|
'seeding',
|
||||||
|
'completed',
|
||||||
|
]);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* POST /api/admin/requests/[id]/retry-download
|
||||||
|
* Retry a failed download for an admin request.
|
||||||
|
*/
|
||||||
|
export async function POST(
|
||||||
|
request: NextRequest,
|
||||||
|
{ params }: { params: Promise<{ id: string }> }
|
||||||
|
) {
|
||||||
|
return requireAuth(request, async (req: AuthenticatedRequest) => {
|
||||||
|
return requireAdmin(req, async () => {
|
||||||
|
try {
|
||||||
|
if (!req.user) {
|
||||||
|
return NextResponse.json(
|
||||||
|
{ error: 'Unauthorized', message: 'User not authenticated' },
|
||||||
|
{ status: 401 }
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
const { id } = await params;
|
||||||
|
|
||||||
|
// Fetch the request with audiobook info
|
||||||
|
const existingRequest = await prisma.request.findFirst({
|
||||||
|
where: { id, deletedAt: null },
|
||||||
|
include: {
|
||||||
|
audiobook: true,
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
if (!existingRequest) {
|
||||||
|
return NextResponse.json(
|
||||||
|
{ error: 'NotFound', message: 'Request not found' },
|
||||||
|
{ status: 404 }
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (existingRequest.status !== 'failed') {
|
||||||
|
return NextResponse.json(
|
||||||
|
{
|
||||||
|
error: 'InvalidStatus',
|
||||||
|
message: `Request is not in a failed state (current status: ${existingRequest.status})`,
|
||||||
|
currentStatus: existingRequest.status,
|
||||||
|
},
|
||||||
|
{ status: 400 }
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Find the most recent selected DownloadHistory record
|
||||||
|
const downloadHistory = await prisma.downloadHistory.findFirst({
|
||||||
|
where: { requestId: id, selected: true },
|
||||||
|
orderBy: { createdAt: 'desc' },
|
||||||
|
});
|
||||||
|
|
||||||
|
if (!downloadHistory) {
|
||||||
|
return NextResponse.json(
|
||||||
|
{
|
||||||
|
error: 'NoHistory',
|
||||||
|
message: 'No previous download attempt found to retry',
|
||||||
|
},
|
||||||
|
{ status: 400 }
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Require a download URL to be able to re-add
|
||||||
|
if (!downloadHistory.magnetLink) {
|
||||||
|
return NextResponse.json(
|
||||||
|
{
|
||||||
|
error: 'NoDownloadUrl',
|
||||||
|
message: 'No download URL available in history to retry',
|
||||||
|
},
|
||||||
|
{ status: 400 }
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
const jobQueue = getJobQueueService();
|
||||||
|
let retryPath: 'resumed_monitoring' | 're_added';
|
||||||
|
|
||||||
|
// Determine if we can attempt to resume monitoring.
|
||||||
|
// downloadClient is stored as a plain string in the DB (can be 'qbittorrent', 'sabnzbd',
|
||||||
|
// 'nzbget', 'transmission', 'deluge', 'direct', or null).
|
||||||
|
const rawClientType: string | null = downloadHistory.downloadClient;
|
||||||
|
const clientId = downloadHistory.downloadClientId;
|
||||||
|
const isDirect = rawClientType === 'direct';
|
||||||
|
|
||||||
|
// Only attempt to query the download client if we have a known DownloadClientType,
|
||||||
|
// a clientId, and it is not a direct (HTTP) download.
|
||||||
|
const canCheckClient = !isDirect && !!rawClientType && !!clientId;
|
||||||
|
// Safe to cast here: we have already confirmed rawClientType is non-null and non-direct
|
||||||
|
const clientType = rawClientType as DownloadClientType | null;
|
||||||
|
|
||||||
|
if (canCheckClient) {
|
||||||
|
// Try to look up the download in the client
|
||||||
|
try {
|
||||||
|
const protocol = CLIENT_PROTOCOL_MAP[clientType as DownloadClientType];
|
||||||
|
const configService = getConfigService();
|
||||||
|
const manager = getDownloadClientManager(configService);
|
||||||
|
const client = await manager.getClientServiceForProtocol(protocol);
|
||||||
|
|
||||||
|
if (client) {
|
||||||
|
const downloadInfo = await client.getDownload(clientId!);
|
||||||
|
|
||||||
|
if (downloadInfo && ALIVE_STATUSES.has(downloadInfo.status)) {
|
||||||
|
// Download is still alive — restart monitoring
|
||||||
|
logger.info(`Retry download: resuming monitoring for request ${id}`, {
|
||||||
|
requestId: id,
|
||||||
|
downloadClientId: clientId,
|
||||||
|
downloadStatus: downloadInfo.status,
|
||||||
|
adminId: req.user.sub,
|
||||||
|
});
|
||||||
|
|
||||||
|
await jobQueue.addMonitorJob(
|
||||||
|
id,
|
||||||
|
downloadHistory.id,
|
||||||
|
clientId!, // canCheckClient guard ensures clientId is non-null
|
||||||
|
clientType as DownloadClientType,
|
||||||
|
0 // no delay — start immediately
|
||||||
|
);
|
||||||
|
|
||||||
|
retryPath = 'resumed_monitoring';
|
||||||
|
} else {
|
||||||
|
// Download not found or is failed — re-add
|
||||||
|
logger.info(`Retry download: download not alive (status: ${downloadInfo?.status ?? 'not found'}), re-adding for request ${id}`, {
|
||||||
|
requestId: id,
|
||||||
|
adminId: req.user.sub,
|
||||||
|
});
|
||||||
|
|
||||||
|
await reAddDownload(jobQueue, id, existingRequest.audiobook, downloadHistory);
|
||||||
|
retryPath = 're_added';
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
// No client configured for that protocol — fall through to re-add
|
||||||
|
logger.warn(`Retry download: no ${protocol} client configured, re-adding for request ${id}`, {
|
||||||
|
requestId: id,
|
||||||
|
adminId: req.user.sub,
|
||||||
|
});
|
||||||
|
|
||||||
|
await reAddDownload(jobQueue, id, existingRequest.audiobook, downloadHistory);
|
||||||
|
retryPath = 're_added';
|
||||||
|
}
|
||||||
|
} catch (clientError) {
|
||||||
|
// Client lookup failed (connection error etc.) — re-add to be safe
|
||||||
|
logger.warn(`Retry download: client check failed, re-adding for request ${id}`, {
|
||||||
|
requestId: id,
|
||||||
|
error: clientError instanceof Error ? clientError.message : String(clientError),
|
||||||
|
adminId: req.user.sub,
|
||||||
|
});
|
||||||
|
|
||||||
|
await reAddDownload(jobQueue, id, existingRequest.audiobook, downloadHistory);
|
||||||
|
retryPath = 're_added';
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
// Direct download (ebook), no clientId, or no clientType — re-add
|
||||||
|
logger.info(`Retry download: re-adding for request ${id} (direct=${isDirect}, hasClientId=${!!clientId})`, {
|
||||||
|
requestId: id,
|
||||||
|
adminId: req.user.sub,
|
||||||
|
});
|
||||||
|
|
||||||
|
await reAddDownload(jobQueue, id, existingRequest.audiobook, downloadHistory);
|
||||||
|
retryPath = 're_added';
|
||||||
|
}
|
||||||
|
|
||||||
|
// Increment downloadAttempts, clear errorMessage, set status to downloading
|
||||||
|
await prisma.request.update({
|
||||||
|
where: { id },
|
||||||
|
data: {
|
||||||
|
status: 'downloading',
|
||||||
|
errorMessage: null,
|
||||||
|
downloadAttempts: { increment: 1 },
|
||||||
|
updatedAt: new Date(),
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
const message =
|
||||||
|
retryPath === 'resumed_monitoring'
|
||||||
|
? 'Download monitoring resumed'
|
||||||
|
: 'Download re-added to client';
|
||||||
|
|
||||||
|
logger.info(`Retry download completed for request ${id} via ${retryPath}`, {
|
||||||
|
requestId: id,
|
||||||
|
adminId: req.user.sub,
|
||||||
|
path: retryPath,
|
||||||
|
});
|
||||||
|
|
||||||
|
return NextResponse.json({
|
||||||
|
success: true,
|
||||||
|
message,
|
||||||
|
path: retryPath,
|
||||||
|
});
|
||||||
|
} catch (error) {
|
||||||
|
logger.error('Failed to retry download', {
|
||||||
|
error: error instanceof Error ? error.message : String(error),
|
||||||
|
});
|
||||||
|
|
||||||
|
return NextResponse.json(
|
||||||
|
{
|
||||||
|
error: 'RetryError',
|
||||||
|
message: 'Failed to retry download',
|
||||||
|
},
|
||||||
|
{ status: 500 }
|
||||||
|
);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Re-add the download to the queue using metadata from DownloadHistory.
|
||||||
|
* Reconstructs a TorrentResult from the stored history fields.
|
||||||
|
*/
|
||||||
|
async function reAddDownload(
|
||||||
|
jobQueue: ReturnType<typeof getJobQueueService>,
|
||||||
|
requestId: string,
|
||||||
|
audiobook: { id: string; title: string; author: string },
|
||||||
|
history: {
|
||||||
|
torrentName: string | null;
|
||||||
|
magnetLink: string | null;
|
||||||
|
indexerName: string;
|
||||||
|
indexerId: number | null;
|
||||||
|
torrentSizeBytes: bigint | null;
|
||||||
|
seeders: number | null;
|
||||||
|
leechers: number | null;
|
||||||
|
torrentHash: string | null;
|
||||||
|
torrentUrl: string | null;
|
||||||
|
}
|
||||||
|
): Promise<void> {
|
||||||
|
const torrent: TorrentResult = {
|
||||||
|
title: history.torrentName ?? audiobook.title,
|
||||||
|
downloadUrl: history.magnetLink!, // Validated non-null before calling this function
|
||||||
|
indexer: history.indexerName,
|
||||||
|
indexerId: history.indexerId ?? undefined,
|
||||||
|
size: history.torrentSizeBytes !== null ? Number(history.torrentSizeBytes) : 0,
|
||||||
|
seeders: history.seeders ?? undefined,
|
||||||
|
leechers: history.leechers ?? undefined,
|
||||||
|
infoHash: history.torrentHash ?? undefined,
|
||||||
|
infoUrl: history.torrentUrl ?? undefined,
|
||||||
|
guid: history.torrentUrl ?? history.magnetLink!,
|
||||||
|
publishDate: new Date(), // Not stored; use current date as a safe default
|
||||||
|
};
|
||||||
|
|
||||||
|
await jobQueue.addDownloadJob(requestId, audiobook, torrent);
|
||||||
|
}
|
||||||
@@ -0,0 +1,133 @@
|
|||||||
|
/**
|
||||||
|
* Component: Admin Custom Search Terms API
|
||||||
|
* Documentation: documentation/admin-dashboard.md
|
||||||
|
*/
|
||||||
|
|
||||||
|
import { NextRequest, NextResponse } from 'next/server';
|
||||||
|
import { requireAuth, requireAdmin, AuthenticatedRequest } from '@/lib/middleware/auth';
|
||||||
|
import { prisma } from '@/lib/db';
|
||||||
|
import { RMABLogger } from '@/lib/utils/logger';
|
||||||
|
|
||||||
|
const logger = RMABLogger.create('API.Admin.SearchTerms');
|
||||||
|
|
||||||
|
/**
|
||||||
|
* PATCH /api/admin/requests/[id]/search-terms
|
||||||
|
* Update custom search terms for a request (admin only)
|
||||||
|
* Body: { searchTerms: string | null, triggerSearch?: boolean }
|
||||||
|
*/
|
||||||
|
export async function PATCH(
|
||||||
|
request: NextRequest,
|
||||||
|
{ params }: { params: Promise<{ id: string }> }
|
||||||
|
) {
|
||||||
|
return requireAuth(request, async (req: AuthenticatedRequest) => {
|
||||||
|
return requireAdmin(req, async () => {
|
||||||
|
try {
|
||||||
|
if (!req.user) {
|
||||||
|
return NextResponse.json(
|
||||||
|
{ error: 'Unauthorized', message: 'User not authenticated' },
|
||||||
|
{ status: 401 }
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
const { id } = await params;
|
||||||
|
|
||||||
|
// Parse body
|
||||||
|
let body;
|
||||||
|
try {
|
||||||
|
body = await req.json();
|
||||||
|
} catch {
|
||||||
|
return NextResponse.json(
|
||||||
|
{ error: 'BadRequest', message: 'Invalid JSON body' },
|
||||||
|
{ status: 400 }
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
const { searchTerms, triggerSearch } = body;
|
||||||
|
|
||||||
|
// Validate searchTerms is string or null
|
||||||
|
if (searchTerms !== null && searchTerms !== undefined && typeof searchTerms !== 'string') {
|
||||||
|
return NextResponse.json(
|
||||||
|
{ error: 'BadRequest', message: 'searchTerms must be a string or null' },
|
||||||
|
{ status: 400 }
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Trim and normalize
|
||||||
|
const normalizedTerms = typeof searchTerms === 'string' ? searchTerms.trim() || null : null;
|
||||||
|
|
||||||
|
// Find the request
|
||||||
|
const existingRequest = await prisma.request.findUnique({
|
||||||
|
where: { id },
|
||||||
|
include: {
|
||||||
|
audiobook: {
|
||||||
|
select: { id: true, title: true, author: true, audibleAsin: true },
|
||||||
|
},
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
if (!existingRequest || existingRequest.deletedAt) {
|
||||||
|
return NextResponse.json(
|
||||||
|
{ error: 'NotFound', message: 'Request not found' },
|
||||||
|
{ status: 404 }
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Update custom search terms
|
||||||
|
await prisma.request.update({
|
||||||
|
where: { id },
|
||||||
|
data: {
|
||||||
|
customSearchTerms: normalizedTerms,
|
||||||
|
updatedAt: new Date(),
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
logger.info(`Custom search terms ${normalizedTerms ? 'set' : 'cleared'} for request ${id}`, {
|
||||||
|
requestId: id,
|
||||||
|
customSearchTerms: normalizedTerms,
|
||||||
|
adminId: req.user.id,
|
||||||
|
});
|
||||||
|
|
||||||
|
// Optionally trigger a new search
|
||||||
|
let searchTriggered = false;
|
||||||
|
if (triggerSearch && ['pending', 'failed', 'awaiting_search'].includes(existingRequest.status)) {
|
||||||
|
// Reset status to pending and clear error
|
||||||
|
await prisma.request.update({
|
||||||
|
where: { id },
|
||||||
|
data: {
|
||||||
|
status: 'pending',
|
||||||
|
errorMessage: null,
|
||||||
|
updatedAt: new Date(),
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
// Queue search job
|
||||||
|
const { getJobQueueService } = await import('@/lib/services/job-queue.service');
|
||||||
|
const jobQueue = getJobQueueService();
|
||||||
|
await jobQueue.addSearchJob(id, {
|
||||||
|
id: existingRequest.audiobook.id,
|
||||||
|
title: existingRequest.audiobook.title,
|
||||||
|
author: existingRequest.audiobook.author,
|
||||||
|
asin: existingRequest.audiobook.audibleAsin || undefined,
|
||||||
|
});
|
||||||
|
|
||||||
|
searchTriggered = true;
|
||||||
|
logger.info(`Search triggered for request ${id} with custom terms`, { requestId: id });
|
||||||
|
}
|
||||||
|
|
||||||
|
return NextResponse.json({
|
||||||
|
success: true,
|
||||||
|
customSearchTerms: normalizedTerms,
|
||||||
|
searchTriggered,
|
||||||
|
});
|
||||||
|
} catch (error) {
|
||||||
|
logger.error('Failed to update search terms', {
|
||||||
|
error: error instanceof Error ? error.message : String(error),
|
||||||
|
});
|
||||||
|
return NextResponse.json(
|
||||||
|
{ error: 'ServerError', message: 'Failed to update search terms' },
|
||||||
|
{ status: 500 }
|
||||||
|
);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
});
|
||||||
|
}
|
||||||
@@ -139,6 +139,8 @@ export async function GET(request: NextRequest) {
|
|||||||
completedAt: request.completedAt,
|
completedAt: request.completedAt,
|
||||||
errorMessage: request.errorMessage,
|
errorMessage: request.errorMessage,
|
||||||
torrentUrl: request.downloadHistory[0]?.torrentUrl || null,
|
torrentUrl: request.downloadHistory[0]?.torrentUrl || null,
|
||||||
|
downloadAttempts: request.downloadAttempts,
|
||||||
|
customSearchTerms: request.customSearchTerms || null,
|
||||||
}));
|
}));
|
||||||
|
|
||||||
return NextResponse.json({
|
return NextResponse.json({
|
||||||
|
|||||||
@@ -0,0 +1,91 @@
|
|||||||
|
/**
|
||||||
|
* Component: Admin Download Access Settings API
|
||||||
|
* Documentation: documentation/settings-pages.md
|
||||||
|
*/
|
||||||
|
|
||||||
|
import { NextRequest, NextResponse } from 'next/server';
|
||||||
|
import { requireAuth, requireAdmin, AuthenticatedRequest } from '@/lib/middleware/auth';
|
||||||
|
import { prisma } from '@/lib/db';
|
||||||
|
import { RMABLogger } from '@/lib/utils/logger';
|
||||||
|
|
||||||
|
const logger = RMABLogger.create('API.Admin.Settings.DownloadAccess');
|
||||||
|
|
||||||
|
const CONFIG_KEY = 'download_access';
|
||||||
|
|
||||||
|
/**
|
||||||
|
* GET /api/admin/settings/download-access
|
||||||
|
* Get current global download access setting
|
||||||
|
*/
|
||||||
|
export async function GET(request: NextRequest) {
|
||||||
|
return requireAuth(request, async (req: AuthenticatedRequest) => {
|
||||||
|
return requireAdmin(req, async () => {
|
||||||
|
try {
|
||||||
|
const config = await prisma.configuration.findUnique({
|
||||||
|
where: { key: CONFIG_KEY },
|
||||||
|
});
|
||||||
|
|
||||||
|
// Default to true if not configured (backward compatibility)
|
||||||
|
const downloadAccess = config === null ? true : config.value === 'true';
|
||||||
|
|
||||||
|
return NextResponse.json({ downloadAccess });
|
||||||
|
} catch (error) {
|
||||||
|
logger.error('Failed to fetch download access setting', {
|
||||||
|
error: error instanceof Error ? error.message : String(error)
|
||||||
|
});
|
||||||
|
return NextResponse.json(
|
||||||
|
{ error: 'Failed to fetch download access setting' },
|
||||||
|
{ status: 500 }
|
||||||
|
);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* PATCH /api/admin/settings/download-access
|
||||||
|
* Update global download access setting
|
||||||
|
*/
|
||||||
|
export async function PATCH(request: NextRequest) {
|
||||||
|
return requireAuth(request, async (req: AuthenticatedRequest) => {
|
||||||
|
return requireAdmin(req, async () => {
|
||||||
|
try {
|
||||||
|
const body = await request.json();
|
||||||
|
const { downloadAccess } = body;
|
||||||
|
|
||||||
|
// Validate input
|
||||||
|
if (typeof downloadAccess !== 'boolean') {
|
||||||
|
return NextResponse.json(
|
||||||
|
{ error: 'Invalid input. downloadAccess must be a boolean' },
|
||||||
|
{ status: 400 }
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Update configuration
|
||||||
|
await prisma.configuration.upsert({
|
||||||
|
where: { key: CONFIG_KEY },
|
||||||
|
create: {
|
||||||
|
key: CONFIG_KEY,
|
||||||
|
value: downloadAccess.toString(),
|
||||||
|
},
|
||||||
|
update: {
|
||||||
|
value: downloadAccess.toString(),
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
logger.info(`Download access setting updated to: ${downloadAccess}`, {
|
||||||
|
userId: req.user?.sub,
|
||||||
|
});
|
||||||
|
|
||||||
|
return NextResponse.json({ downloadAccess });
|
||||||
|
} catch (error) {
|
||||||
|
logger.error('Failed to update download access setting', {
|
||||||
|
error: error instanceof Error ? error.message : String(error)
|
||||||
|
});
|
||||||
|
return NextResponse.json(
|
||||||
|
{ error: 'Failed to update download access setting' },
|
||||||
|
{ status: 500 }
|
||||||
|
);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
});
|
||||||
|
}
|
||||||
@@ -78,7 +78,7 @@ export async function PUT(request: NextRequest) {
|
|||||||
// Anna's Archive specific settings
|
// Anna's Archive specific settings
|
||||||
{
|
{
|
||||||
key: 'ebook_sidecar_base_url',
|
key: 'ebook_sidecar_base_url',
|
||||||
value: baseUrl || 'https://annas-archive.li',
|
value: baseUrl || 'https://annas-archive.gl',
|
||||||
category: 'ebook',
|
category: 'ebook',
|
||||||
description: 'Base URL for Anna\'s Archive',
|
description: 'Base URL for Anna\'s Archive',
|
||||||
},
|
},
|
||||||
|
|||||||
@@ -15,7 +15,7 @@ export async function PUT(request: NextRequest) {
|
|||||||
return requireAuth(request, async (req: AuthenticatedRequest) => {
|
return requireAuth(request, async (req: AuthenticatedRequest) => {
|
||||||
return requireAdmin(req, async () => {
|
return requireAdmin(req, async () => {
|
||||||
try {
|
try {
|
||||||
const { downloadDir, mediaDir, audiobookPathTemplate, ebookPathTemplate, metadataTaggingEnabled, chapterMergingEnabled } = await request.json();
|
const { downloadDir, mediaDir, audiobookPathTemplate, ebookPathTemplate, metadataTaggingEnabled, chapterMergingEnabled, fileRenameEnabled, fileRenameTemplate } = await request.json();
|
||||||
|
|
||||||
if (!downloadDir || !mediaDir) {
|
if (!downloadDir || !mediaDir) {
|
||||||
return NextResponse.json(
|
return NextResponse.json(
|
||||||
@@ -97,6 +97,32 @@ export async function PUT(request: NextRequest) {
|
|||||||
},
|
},
|
||||||
});
|
});
|
||||||
|
|
||||||
|
// Update file rename setting
|
||||||
|
await prisma.configuration.upsert({
|
||||||
|
where: { key: 'file_rename_enabled' },
|
||||||
|
update: { value: String(fileRenameEnabled ?? false) },
|
||||||
|
create: {
|
||||||
|
key: 'file_rename_enabled',
|
||||||
|
value: String(fileRenameEnabled ?? false),
|
||||||
|
category: 'automation',
|
||||||
|
description: 'Rename audio and ebook files using a custom naming template during organization',
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
// Update file rename template
|
||||||
|
if (fileRenameTemplate !== undefined) {
|
||||||
|
await prisma.configuration.upsert({
|
||||||
|
where: { key: 'file_rename_template' },
|
||||||
|
update: { value: fileRenameTemplate },
|
||||||
|
create: {
|
||||||
|
key: 'file_rename_template',
|
||||||
|
value: fileRenameTemplate,
|
||||||
|
category: 'automation',
|
||||||
|
description: 'Template for renaming audio and ebook files during organization',
|
||||||
|
},
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
logger.info('Paths settings updated');
|
logger.info('Paths settings updated');
|
||||||
|
|
||||||
// Clear config cache for all updated keys so services get fresh values
|
// Clear config cache for all updated keys so services get fresh values
|
||||||
@@ -107,6 +133,8 @@ export async function PUT(request: NextRequest) {
|
|||||||
configService.clearCache('ebook_path_template');
|
configService.clearCache('ebook_path_template');
|
||||||
configService.clearCache('metadata_tagging_enabled');
|
configService.clearCache('metadata_tagging_enabled');
|
||||||
configService.clearCache('chapter_merging_enabled');
|
configService.clearCache('chapter_merging_enabled');
|
||||||
|
configService.clearCache('file_rename_enabled');
|
||||||
|
configService.clearCache('file_rename_template');
|
||||||
|
|
||||||
// Invalidate all download client singletons to force reload of download_dir
|
// Invalidate all download client singletons to force reload of download_dir
|
||||||
const { invalidateDownloadClientManager } = await import('@/lib/services/download-client-manager.service');
|
const { invalidateDownloadClientManager } = await import('@/lib/services/download-client-manager.service');
|
||||||
|
|||||||
@@ -128,6 +128,8 @@ export async function GET(request: NextRequest) {
|
|||||||
ebookPathTemplate: configMap.get('ebook_path_template') || configMap.get('audiobook_path_template') || '{author}/{title} {asin}',
|
ebookPathTemplate: configMap.get('ebook_path_template') || configMap.get('audiobook_path_template') || '{author}/{title} {asin}',
|
||||||
metadataTaggingEnabled: configMap.get('metadata_tagging_enabled') === 'true',
|
metadataTaggingEnabled: configMap.get('metadata_tagging_enabled') === 'true',
|
||||||
chapterMergingEnabled: configMap.get('chapter_merging_enabled') === 'true',
|
chapterMergingEnabled: configMap.get('chapter_merging_enabled') === 'true',
|
||||||
|
fileRenameEnabled: configMap.get('file_rename_enabled') === 'true',
|
||||||
|
fileRenameTemplate: configMap.get('file_rename_template') || '{title}',
|
||||||
},
|
},
|
||||||
ebook: {
|
ebook: {
|
||||||
// New granular source toggles (with migration from legacy ebook_sidecar_enabled)
|
// New granular source toggles (with migration from legacy ebook_sidecar_enabled)
|
||||||
@@ -136,7 +138,7 @@ export async function GET(request: NextRequest) {
|
|||||||
(configMap.get('ebook_annas_archive_enabled') === undefined && configMap.get('ebook_sidecar_enabled') === 'true'),
|
(configMap.get('ebook_annas_archive_enabled') === undefined && configMap.get('ebook_sidecar_enabled') === 'true'),
|
||||||
indexerSearchEnabled: configMap.get('ebook_indexer_search_enabled') === 'true',
|
indexerSearchEnabled: configMap.get('ebook_indexer_search_enabled') === 'true',
|
||||||
// Anna's Archive specific settings
|
// Anna's Archive specific settings
|
||||||
baseUrl: configMap.get('ebook_sidecar_base_url') || 'https://annas-archive.li',
|
baseUrl: configMap.get('ebook_sidecar_base_url') || 'https://annas-archive.gl',
|
||||||
flaresolverrUrl: configMap.get('ebook_sidecar_flaresolverr_url') || '',
|
flaresolverrUrl: configMap.get('ebook_sidecar_flaresolverr_url') || '',
|
||||||
// General settings
|
// General settings
|
||||||
preferredFormat: configMap.get('ebook_sidecar_preferred_format') || 'epub',
|
preferredFormat: configMap.get('ebook_sidecar_preferred_format') || 'epub',
|
||||||
|
|||||||
@@ -19,7 +19,7 @@ export async function PUT(
|
|||||||
try {
|
try {
|
||||||
const { id } = await params;
|
const { id } = await params;
|
||||||
const body = await request.json();
|
const body = await request.json();
|
||||||
const { role, autoApproveRequests, interactiveSearchAccess } = body;
|
const { role, autoApproveRequests, interactiveSearchAccess, downloadAccess } = body;
|
||||||
|
|
||||||
// Validate role
|
// Validate role
|
||||||
if (!role || (role !== 'user' && role !== 'admin')) {
|
if (!role || (role !== 'user' && role !== 'admin')) {
|
||||||
@@ -45,6 +45,14 @@ export async function PUT(
|
|||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Validate downloadAccess (optional)
|
||||||
|
if (downloadAccess !== undefined && downloadAccess !== null && typeof downloadAccess !== 'boolean') {
|
||||||
|
return NextResponse.json(
|
||||||
|
{ error: 'Invalid downloadAccess. Must be a boolean or null' },
|
||||||
|
{ status: 400 }
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
// Prevent user from demoting themselves
|
// Prevent user from demoting themselves
|
||||||
if (req.user && id === req.user.sub) {
|
if (req.user && id === req.user.sub) {
|
||||||
return NextResponse.json(
|
return NextResponse.json(
|
||||||
@@ -112,15 +120,24 @@ export async function PUT(
|
|||||||
{ status: 400 }
|
{ status: 400 }
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
if (role === 'admin' && downloadAccess === false) {
|
||||||
|
return NextResponse.json(
|
||||||
|
{ error: 'Admins always have download access. Cannot set downloadAccess to false for admin users.' },
|
||||||
|
{ status: 400 }
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
// Prepare update data
|
// Prepare update data
|
||||||
const updateData: { role: string; autoApproveRequests?: boolean | null; interactiveSearchAccess?: boolean | null } = { role };
|
const updateData: { role: string; autoApproveRequests?: boolean | null; interactiveSearchAccess?: boolean | null; downloadAccess?: boolean | null } = { role };
|
||||||
if (autoApproveRequests !== undefined) {
|
if (autoApproveRequests !== undefined) {
|
||||||
updateData.autoApproveRequests = autoApproveRequests;
|
updateData.autoApproveRequests = autoApproveRequests;
|
||||||
}
|
}
|
||||||
if (interactiveSearchAccess !== undefined) {
|
if (interactiveSearchAccess !== undefined) {
|
||||||
updateData.interactiveSearchAccess = interactiveSearchAccess;
|
updateData.interactiveSearchAccess = interactiveSearchAccess;
|
||||||
}
|
}
|
||||||
|
if (downloadAccess !== undefined) {
|
||||||
|
updateData.downloadAccess = downloadAccess;
|
||||||
|
}
|
||||||
|
|
||||||
// Update user
|
// Update user
|
||||||
const updatedUser = await prisma.user.update({
|
const updatedUser = await prisma.user.update({
|
||||||
@@ -132,6 +149,7 @@ export async function PUT(
|
|||||||
role: true,
|
role: true,
|
||||||
autoApproveRequests: true,
|
autoApproveRequests: true,
|
||||||
interactiveSearchAccess: true,
|
interactiveSearchAccess: true,
|
||||||
|
downloadAccess: true,
|
||||||
},
|
},
|
||||||
});
|
});
|
||||||
|
|
||||||
|
|||||||
@@ -32,6 +32,7 @@ export async function GET(request: NextRequest) {
|
|||||||
lastLoginAt: true,
|
lastLoginAt: true,
|
||||||
autoApproveRequests: true,
|
autoApproveRequests: true,
|
||||||
interactiveSearchAccess: true,
|
interactiveSearchAccess: true,
|
||||||
|
downloadAccess: true,
|
||||||
_count: {
|
_count: {
|
||||||
select: {
|
select: {
|
||||||
requests: true,
|
requests: true,
|
||||||
|
|||||||
@@ -0,0 +1,39 @@
|
|||||||
|
/**
|
||||||
|
* Component: Audible Categories API Route
|
||||||
|
* Documentation: documentation/features/home-sections.md
|
||||||
|
*
|
||||||
|
* Live scrape of top-level Audible categories for the home section config modal.
|
||||||
|
*/
|
||||||
|
|
||||||
|
import { NextRequest, NextResponse } from 'next/server';
|
||||||
|
import { requireAuth, AuthenticatedRequest } from '@/lib/middleware/auth';
|
||||||
|
import { RMABLogger } from '@/lib/utils/logger';
|
||||||
|
|
||||||
|
const logger = RMABLogger.create('API.Audible.Categories');
|
||||||
|
|
||||||
|
/**
|
||||||
|
* GET /api/audible/categories
|
||||||
|
* Returns top-level Audible categories scraped live from audible.com/categories
|
||||||
|
*/
|
||||||
|
export async function GET(request: NextRequest) {
|
||||||
|
return requireAuth(request, async (_req: AuthenticatedRequest) => {
|
||||||
|
try {
|
||||||
|
const { getAudibleService } = await import('@/lib/integrations/audible.service');
|
||||||
|
const audibleService = getAudibleService();
|
||||||
|
const categories = await audibleService.getCategories();
|
||||||
|
|
||||||
|
return NextResponse.json({
|
||||||
|
success: true,
|
||||||
|
categories,
|
||||||
|
});
|
||||||
|
} catch (error) {
|
||||||
|
logger.error('Failed to fetch categories', {
|
||||||
|
error: error instanceof Error ? error.message : String(error),
|
||||||
|
});
|
||||||
|
return NextResponse.json(
|
||||||
|
{ error: 'FetchError', message: 'Failed to fetch Audible categories' },
|
||||||
|
{ status: 500 }
|
||||||
|
);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
@@ -0,0 +1,70 @@
|
|||||||
|
/**
|
||||||
|
* Component: Audiobook Download Status API Route
|
||||||
|
* Documentation: documentation/backend/api.md
|
||||||
|
*
|
||||||
|
* Returns whether a downloadable file exists for this audiobook (by ASIN).
|
||||||
|
* Used by AudiobookDetailsModal to show the download link regardless of context.
|
||||||
|
*/
|
||||||
|
|
||||||
|
import { NextRequest, NextResponse } from 'next/server';
|
||||||
|
import { requireAuth, AuthenticatedRequest } from '@/lib/middleware/auth';
|
||||||
|
import { prisma } from '@/lib/db';
|
||||||
|
import { COMPLETED_STATUSES } from '@/lib/constants/request-statuses';
|
||||||
|
import { resolveDownloadAccess } from '@/lib/utils/permissions';
|
||||||
|
|
||||||
|
/**
|
||||||
|
* GET /api/audiobooks/[asin]/download-status
|
||||||
|
* Returns { downloadAvailable, requestId } for the current user's completed request.
|
||||||
|
*/
|
||||||
|
export async function GET(
|
||||||
|
request: NextRequest,
|
||||||
|
{ params }: { params: Promise<{ asin: string }> }
|
||||||
|
) {
|
||||||
|
return requireAuth(request, async (req: AuthenticatedRequest) => {
|
||||||
|
if (!req.user) {
|
||||||
|
return NextResponse.json({ error: 'Unauthorized' }, { status: 401 });
|
||||||
|
}
|
||||||
|
|
||||||
|
// Check download permission - if denied, don't reveal file existence
|
||||||
|
const userRecord = await prisma.user.findUnique({
|
||||||
|
where: { id: req.user.id },
|
||||||
|
select: { role: true, downloadAccess: true },
|
||||||
|
});
|
||||||
|
const hasDownloadAccess = await resolveDownloadAccess(
|
||||||
|
userRecord?.role ?? 'user',
|
||||||
|
userRecord?.downloadAccess ?? null
|
||||||
|
);
|
||||||
|
if (!hasDownloadAccess) {
|
||||||
|
return NextResponse.json({ downloadAvailable: false, requestId: null });
|
||||||
|
}
|
||||||
|
|
||||||
|
const { asin } = await params;
|
||||||
|
|
||||||
|
const audiobook = await prisma.audiobook.findFirst({
|
||||||
|
where: { audibleAsin: asin },
|
||||||
|
select: { id: true, filePath: true },
|
||||||
|
});
|
||||||
|
|
||||||
|
if (!audiobook) {
|
||||||
|
return NextResponse.json({ downloadAvailable: false, requestId: null });
|
||||||
|
}
|
||||||
|
|
||||||
|
// Find any completed request for this audiobook that has a file
|
||||||
|
const completedRequest = await prisma.request.findFirst({
|
||||||
|
where: {
|
||||||
|
audiobookId: audiobook.id,
|
||||||
|
status: { in: [...COMPLETED_STATUSES] },
|
||||||
|
deletedAt: null,
|
||||||
|
},
|
||||||
|
select: { id: true },
|
||||||
|
orderBy: { createdAt: 'desc' },
|
||||||
|
});
|
||||||
|
|
||||||
|
const downloadAvailable = !!completedRequest && !!audiobook.filePath;
|
||||||
|
|
||||||
|
return NextResponse.json({
|
||||||
|
downloadAvailable,
|
||||||
|
requestId: downloadAvailable ? completedRequest!.id : null,
|
||||||
|
});
|
||||||
|
});
|
||||||
|
}
|
||||||
@@ -227,7 +227,7 @@ export async function POST(
|
|||||||
const isAnnasArchiveEnabled = annasArchiveEnabled === 'true';
|
const isAnnasArchiveEnabled = annasArchiveEnabled === 'true';
|
||||||
const isIndexerSearchEnabled = indexerSearchEnabled === 'true';
|
const isIndexerSearchEnabled = indexerSearchEnabled === 'true';
|
||||||
const format = preferredFormat || 'epub';
|
const format = preferredFormat || 'epub';
|
||||||
const annasBaseUrl = baseUrl || 'https://annas-archive.li';
|
const annasBaseUrl = baseUrl || 'https://annas-archive.gl';
|
||||||
|
|
||||||
// Get language code from Audible region config
|
// Get language code from Audible region config
|
||||||
const region = await configService.getAudibleRegion() as AudibleRegion;
|
const region = await configService.getAudibleRegion() as AudibleRegion;
|
||||||
|
|||||||
@@ -0,0 +1,154 @@
|
|||||||
|
/**
|
||||||
|
* Component: Category Audiobooks API Route
|
||||||
|
* Documentation: documentation/features/home-sections.md
|
||||||
|
*
|
||||||
|
* Serves audiobooks for a specific Audible category from AudibleCacheCategory,
|
||||||
|
* with the same enrichment pattern as popular/new-releases routes.
|
||||||
|
*/
|
||||||
|
|
||||||
|
import { NextRequest, NextResponse } from 'next/server';
|
||||||
|
import { prisma } from '@/lib/db';
|
||||||
|
import { enrichAudiobooksWithMatches, getAvailableAsins } from '@/lib/utils/audiobook-matcher';
|
||||||
|
import { getCurrentUser } from '@/lib/middleware/auth';
|
||||||
|
import { RMABLogger } from '@/lib/utils/logger';
|
||||||
|
|
||||||
|
const logger = RMABLogger.create('API.Audiobooks.Category');
|
||||||
|
|
||||||
|
/**
|
||||||
|
* GET /api/audiobooks/category/[categoryId]?page=1&limit=20&hideAvailable=false
|
||||||
|
*/
|
||||||
|
export async function GET(
|
||||||
|
request: NextRequest,
|
||||||
|
{ params }: { params: Promise<{ categoryId: string }> }
|
||||||
|
) {
|
||||||
|
try {
|
||||||
|
const { categoryId } = await params;
|
||||||
|
const searchParams = request.nextUrl.searchParams;
|
||||||
|
const page = parseInt(searchParams.get('page') || '1', 10);
|
||||||
|
const limit = parseInt(searchParams.get('limit') || '20', 10);
|
||||||
|
const hideAvailable = searchParams.get('hideAvailable') === 'true';
|
||||||
|
|
||||||
|
if (page < 1 || limit < 1 || limit > 100) {
|
||||||
|
return NextResponse.json(
|
||||||
|
{ error: 'ValidationError', message: 'Invalid pagination parameters.' },
|
||||||
|
{ status: 400 }
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
const skip = (page - 1) * limit;
|
||||||
|
|
||||||
|
// Get excluded ASINs when hideAvailable
|
||||||
|
let excludedAsins: string[] = [];
|
||||||
|
if (hideAvailable) {
|
||||||
|
const availableSet = await getAvailableAsins();
|
||||||
|
excludedAsins = [...availableSet];
|
||||||
|
}
|
||||||
|
|
||||||
|
// Query AudibleCacheCategory joined with AudibleCache
|
||||||
|
const whereClause: any = { categoryId };
|
||||||
|
if (excludedAsins.length > 0) {
|
||||||
|
whereClause.asin = { notIn: excludedAsins };
|
||||||
|
}
|
||||||
|
|
||||||
|
const [categoryEntries, totalCount] = await Promise.all([
|
||||||
|
prisma.audibleCacheCategory.findMany({
|
||||||
|
where: whereClause,
|
||||||
|
orderBy: { rank: 'asc' },
|
||||||
|
skip,
|
||||||
|
take: limit,
|
||||||
|
select: { asin: true, rank: true },
|
||||||
|
}),
|
||||||
|
prisma.audibleCacheCategory.count({ where: whereClause }),
|
||||||
|
]);
|
||||||
|
|
||||||
|
if (totalCount === 0) {
|
||||||
|
return NextResponse.json({
|
||||||
|
success: true,
|
||||||
|
audiobooks: [],
|
||||||
|
count: 0,
|
||||||
|
totalCount: 0,
|
||||||
|
page,
|
||||||
|
totalPages: 0,
|
||||||
|
hasMore: false,
|
||||||
|
message: 'No audiobooks found for this category. Data may not have been refreshed yet.',
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
// Fetch full metadata from AudibleCache for these ASINs
|
||||||
|
const asins = categoryEntries.map((e) => e.asin);
|
||||||
|
const cacheEntries = await prisma.audibleCache.findMany({
|
||||||
|
where: { asin: { in: asins } },
|
||||||
|
select: {
|
||||||
|
asin: true,
|
||||||
|
title: true,
|
||||||
|
author: true,
|
||||||
|
narrator: true,
|
||||||
|
description: true,
|
||||||
|
coverArtUrl: true,
|
||||||
|
cachedCoverPath: true,
|
||||||
|
durationMinutes: true,
|
||||||
|
releaseDate: true,
|
||||||
|
rating: true,
|
||||||
|
genres: true,
|
||||||
|
lastSyncedAt: true,
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
// Build a map for ordering by rank
|
||||||
|
const cacheMap = new Map(cacheEntries.map((e) => [e.asin, e]));
|
||||||
|
|
||||||
|
// Transform to matcher input format, preserving rank order
|
||||||
|
const audibleBooks = categoryEntries
|
||||||
|
.map((entry) => {
|
||||||
|
const book = cacheMap.get(entry.asin);
|
||||||
|
if (!book) return null;
|
||||||
|
|
||||||
|
let coverUrl = book.coverArtUrl || undefined;
|
||||||
|
if (book.cachedCoverPath) {
|
||||||
|
const filename = book.cachedCoverPath.split('/').pop();
|
||||||
|
coverUrl = `/api/cache/thumbnails/${filename}`;
|
||||||
|
}
|
||||||
|
|
||||||
|
return {
|
||||||
|
asin: book.asin,
|
||||||
|
title: book.title,
|
||||||
|
author: book.author,
|
||||||
|
narrator: book.narrator || undefined,
|
||||||
|
description: book.description || undefined,
|
||||||
|
coverArtUrl: coverUrl,
|
||||||
|
durationMinutes: book.durationMinutes || undefined,
|
||||||
|
releaseDate: book.releaseDate?.toISOString() || undefined,
|
||||||
|
rating: book.rating ? parseFloat(book.rating.toString()) : undefined,
|
||||||
|
genres: (book.genres as string[]) || [],
|
||||||
|
};
|
||||||
|
})
|
||||||
|
.filter(Boolean) as any[];
|
||||||
|
|
||||||
|
// Enrich with library matching and request status
|
||||||
|
const currentUser = getCurrentUser(request);
|
||||||
|
const userId = currentUser?.sub || undefined;
|
||||||
|
const enrichedAudiobooks = await enrichAudiobooksWithMatches(audibleBooks, userId);
|
||||||
|
|
||||||
|
const totalPages = Math.ceil(totalCount / limit);
|
||||||
|
const hasMore = page < totalPages;
|
||||||
|
|
||||||
|
return NextResponse.json({
|
||||||
|
success: true,
|
||||||
|
audiobooks: enrichedAudiobooks,
|
||||||
|
count: enrichedAudiobooks.length,
|
||||||
|
totalCount,
|
||||||
|
page,
|
||||||
|
totalPages,
|
||||||
|
hasMore,
|
||||||
|
lastSync: cacheEntries[0]?.lastSyncedAt?.toISOString() || null,
|
||||||
|
});
|
||||||
|
} catch (error) {
|
||||||
|
logger.error('Failed to get category audiobooks', {
|
||||||
|
error: error instanceof Error ? error.message : String(error),
|
||||||
|
});
|
||||||
|
return NextResponse.json(
|
||||||
|
{ error: 'FetchError', message: 'Failed to fetch category audiobooks' },
|
||||||
|
{ status: 500 }
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -2,12 +2,14 @@
|
|||||||
* Component: Audiobook Covers API Route
|
* Component: Audiobook Covers API Route
|
||||||
* Documentation: documentation/frontend/pages/login.md
|
* Documentation: documentation/frontend/pages/login.md
|
||||||
*
|
*
|
||||||
* Serves random popular audiobook covers for login page floating animations
|
* Serves random popular audiobook covers for login page floating animations.
|
||||||
|
* Queries AudibleCacheCategory with '__popular__' categoryId for cover sources.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
import { NextResponse } from 'next/server';
|
import { NextResponse } from 'next/server';
|
||||||
import { prisma } from '@/lib/db';
|
import { prisma } from '@/lib/db';
|
||||||
import { RMABLogger } from '@/lib/utils/logger';
|
import { RMABLogger } from '@/lib/utils/logger';
|
||||||
|
import { POPULAR_CATEGORY_ID } from '@/lib/processors/audible-refresh.processor';
|
||||||
|
|
||||||
const logger = RMABLogger.create('API.Audiobooks.Covers');
|
const logger = RMABLogger.create('API.Audiobooks.Covers');
|
||||||
|
|
||||||
@@ -20,18 +22,22 @@ const logger = RMABLogger.create('API.Audiobooks.Covers');
|
|||||||
*/
|
*/
|
||||||
export async function GET() {
|
export async function GET() {
|
||||||
try {
|
try {
|
||||||
// Fetch all popular audiobooks with covers (up to 200)
|
// Get popular ASINs from category table (up to 200)
|
||||||
|
const categoryEntries = await prisma.audibleCacheCategory.findMany({
|
||||||
|
where: { categoryId: POPULAR_CATEGORY_ID },
|
||||||
|
orderBy: { rank: 'asc' },
|
||||||
|
take: 200,
|
||||||
|
select: { asin: true },
|
||||||
|
});
|
||||||
|
|
||||||
|
const asins = categoryEntries.map((e) => e.asin);
|
||||||
|
|
||||||
|
// Fetch cover data from AudibleCache for popular ASINs with cached covers
|
||||||
const audiobooks = await prisma.audibleCache.findMany({
|
const audiobooks = await prisma.audibleCache.findMany({
|
||||||
where: {
|
where: {
|
||||||
isPopular: true,
|
asin: { in: asins },
|
||||||
cachedCoverPath: {
|
cachedCoverPath: { not: null },
|
||||||
not: null,
|
|
||||||
},
|
|
||||||
},
|
},
|
||||||
orderBy: {
|
|
||||||
popularRank: 'asc',
|
|
||||||
},
|
|
||||||
take: 200,
|
|
||||||
select: {
|
select: {
|
||||||
asin: true,
|
asin: true,
|
||||||
title: true,
|
title: true,
|
||||||
|
|||||||
@@ -2,20 +2,22 @@
|
|||||||
* Component: New Releases API Route
|
* Component: New Releases API Route
|
||||||
* Documentation: documentation/integrations/audible.md
|
* Documentation: documentation/integrations/audible.md
|
||||||
*
|
*
|
||||||
* Serves new release audiobooks from audible_cache with real-time Plex matching
|
* Serves new release audiobooks from AudibleCacheCategory with real-time library matching.
|
||||||
|
* New releases are stored with categoryId '__new_releases__' in the unified category table.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
import { NextRequest, NextResponse } from 'next/server';
|
import { NextRequest, NextResponse } from 'next/server';
|
||||||
import { prisma } from '@/lib/db';
|
import { prisma } from '@/lib/db';
|
||||||
import { enrichAudiobooksWithMatches } from '@/lib/utils/audiobook-matcher';
|
import { enrichAudiobooksWithMatches, getAvailableAsins } from '@/lib/utils/audiobook-matcher';
|
||||||
import { getCurrentUser } from '@/lib/middleware/auth';
|
import { getCurrentUser } from '@/lib/middleware/auth';
|
||||||
import { RMABLogger } from '@/lib/utils/logger';
|
import { RMABLogger } from '@/lib/utils/logger';
|
||||||
|
import { NEW_RELEASES_CATEGORY_ID } from '@/lib/processors/audible-refresh.processor';
|
||||||
|
|
||||||
const logger = RMABLogger.create('API.Audiobooks.NewReleases');
|
const logger = RMABLogger.create('API.Audiobooks.NewReleases');
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* GET /api/audiobooks/new-releases?page=1&limit=20
|
* GET /api/audiobooks/new-releases?page=1&limit=20
|
||||||
* Get new release audiobooks from audible_cache with pagination
|
* Get new release audiobooks from AudibleCacheCategory with pagination
|
||||||
*
|
*
|
||||||
* Real-time matching against plex_library determines availability.
|
* Real-time matching against plex_library determines availability.
|
||||||
*/
|
*/
|
||||||
@@ -24,6 +26,7 @@ export async function GET(request: NextRequest) {
|
|||||||
const searchParams = request.nextUrl.searchParams;
|
const searchParams = request.nextUrl.searchParams;
|
||||||
const page = parseInt(searchParams.get('page') || '1', 10);
|
const page = parseInt(searchParams.get('page') || '1', 10);
|
||||||
const limit = parseInt(searchParams.get('limit') || '20', 10);
|
const limit = parseInt(searchParams.get('limit') || '20', 10);
|
||||||
|
const hideAvailable = searchParams.get('hideAvailable') === 'true';
|
||||||
|
|
||||||
// Validate pagination parameters
|
// Validate pagination parameters
|
||||||
if (page < 1 || limit < 1 || limit > 100) {
|
if (page < 1 || limit < 1 || limit > 100) {
|
||||||
@@ -38,38 +41,28 @@ export async function GET(request: NextRequest) {
|
|||||||
|
|
||||||
const skip = (page - 1) * limit;
|
const skip = (page - 1) * limit;
|
||||||
|
|
||||||
// Query audible_cache for new release audiobooks
|
// When hideAvailable is enabled, exclude ASINs that are in the library or have completed requests
|
||||||
const [audiobooks, totalCount] = await Promise.all([
|
let excludedAsins: string[] = [];
|
||||||
prisma.audibleCache.findMany({
|
if (hideAvailable) {
|
||||||
where: {
|
const availableSet = await getAvailableAsins();
|
||||||
isNewRelease: true,
|
excludedAsins = [...availableSet];
|
||||||
},
|
}
|
||||||
orderBy: {
|
|
||||||
newReleaseRank: 'asc',
|
const whereClause: any = { categoryId: NEW_RELEASES_CATEGORY_ID };
|
||||||
},
|
if (excludedAsins.length > 0) {
|
||||||
|
whereClause.asin = { notIn: excludedAsins };
|
||||||
|
}
|
||||||
|
|
||||||
|
// Query AudibleCacheCategory for new release audiobooks
|
||||||
|
const [categoryEntries, totalCount] = await Promise.all([
|
||||||
|
prisma.audibleCacheCategory.findMany({
|
||||||
|
where: whereClause,
|
||||||
|
orderBy: { rank: 'asc' },
|
||||||
skip,
|
skip,
|
||||||
take: limit,
|
take: limit,
|
||||||
select: {
|
select: { asin: true, rank: true },
|
||||||
id: true,
|
|
||||||
asin: true,
|
|
||||||
title: true,
|
|
||||||
author: true,
|
|
||||||
narrator: true,
|
|
||||||
description: true,
|
|
||||||
coverArtUrl: true,
|
|
||||||
cachedCoverPath: true,
|
|
||||||
durationMinutes: true,
|
|
||||||
releaseDate: true,
|
|
||||||
rating: true,
|
|
||||||
genres: true,
|
|
||||||
lastSyncedAt: true,
|
|
||||||
},
|
|
||||||
}),
|
|
||||||
prisma.audibleCache.count({
|
|
||||||
where: {
|
|
||||||
isNewRelease: true,
|
|
||||||
},
|
|
||||||
}),
|
}),
|
||||||
|
prisma.audibleCacheCategory.count({ where: whereClause }),
|
||||||
]);
|
]);
|
||||||
|
|
||||||
// If no data found, return helpful message
|
// If no data found, return helpful message
|
||||||
@@ -86,30 +79,56 @@ export async function GET(request: NextRequest) {
|
|||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
// Transform to matcher input format (uses ASIN as required field)
|
// Fetch full metadata from AudibleCache for these ASINs
|
||||||
// Use cached cover path when available, otherwise fall back to coverArtUrl
|
const asins = categoryEntries.map((e) => e.asin);
|
||||||
const audibleBooks = audiobooks.map((book) => {
|
const cacheEntries = await prisma.audibleCache.findMany({
|
||||||
// Convert cached path to API URL if it exists
|
where: { asin: { in: asins } },
|
||||||
let coverUrl = book.coverArtUrl || undefined;
|
select: {
|
||||||
if (book.cachedCoverPath) {
|
asin: true,
|
||||||
const filename = book.cachedCoverPath.split('/').pop();
|
title: true,
|
||||||
coverUrl = `/api/cache/thumbnails/${filename}`;
|
author: true,
|
||||||
}
|
narrator: true,
|
||||||
|
description: true,
|
||||||
return {
|
coverArtUrl: true,
|
||||||
asin: book.asin,
|
cachedCoverPath: true,
|
||||||
title: book.title,
|
durationMinutes: true,
|
||||||
author: book.author,
|
releaseDate: true,
|
||||||
narrator: book.narrator || undefined,
|
rating: true,
|
||||||
description: book.description || undefined,
|
genres: true,
|
||||||
coverArtUrl: coverUrl,
|
lastSyncedAt: true,
|
||||||
durationMinutes: book.durationMinutes || undefined,
|
},
|
||||||
releaseDate: book.releaseDate?.toISOString() || undefined,
|
|
||||||
rating: book.rating ? parseFloat(book.rating.toString()) : undefined,
|
|
||||||
genres: (book.genres as string[]) || [],
|
|
||||||
};
|
|
||||||
});
|
});
|
||||||
|
|
||||||
|
// Build a map for ordering by rank
|
||||||
|
const cacheMap = new Map(cacheEntries.map((e) => [e.asin, e]));
|
||||||
|
|
||||||
|
// Transform to matcher input format, preserving rank order
|
||||||
|
const audibleBooks = categoryEntries
|
||||||
|
.map((entry) => {
|
||||||
|
const book = cacheMap.get(entry.asin);
|
||||||
|
if (!book) return null;
|
||||||
|
|
||||||
|
let coverUrl = book.coverArtUrl || undefined;
|
||||||
|
if (book.cachedCoverPath) {
|
||||||
|
const filename = book.cachedCoverPath.split('/').pop();
|
||||||
|
coverUrl = `/api/cache/thumbnails/${filename}`;
|
||||||
|
}
|
||||||
|
|
||||||
|
return {
|
||||||
|
asin: book.asin,
|
||||||
|
title: book.title,
|
||||||
|
author: book.author,
|
||||||
|
narrator: book.narrator || undefined,
|
||||||
|
description: book.description || undefined,
|
||||||
|
coverArtUrl: coverUrl,
|
||||||
|
durationMinutes: book.durationMinutes || undefined,
|
||||||
|
releaseDate: book.releaseDate?.toISOString() || undefined,
|
||||||
|
rating: book.rating ? parseFloat(book.rating.toString()) : undefined,
|
||||||
|
genres: (book.genres as string[]) || [],
|
||||||
|
};
|
||||||
|
})
|
||||||
|
.filter(Boolean) as any[];
|
||||||
|
|
||||||
// Get current user (optional - for request status enrichment)
|
// Get current user (optional - for request status enrichment)
|
||||||
const currentUser = getCurrentUser(request);
|
const currentUser = getCurrentUser(request);
|
||||||
const userId = currentUser?.sub || undefined;
|
const userId = currentUser?.sub || undefined;
|
||||||
@@ -128,7 +147,7 @@ export async function GET(request: NextRequest) {
|
|||||||
page,
|
page,
|
||||||
totalPages,
|
totalPages,
|
||||||
hasMore,
|
hasMore,
|
||||||
lastSync: audiobooks[0]?.lastSyncedAt?.toISOString() || null,
|
lastSync: cacheEntries[0]?.lastSyncedAt?.toISOString() || null,
|
||||||
});
|
});
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
logger.error('Failed to get new releases', { error: error instanceof Error ? error.message : String(error) });
|
logger.error('Failed to get new releases', { error: error instanceof Error ? error.message : String(error) });
|
||||||
|
|||||||
@@ -2,20 +2,22 @@
|
|||||||
* Component: Popular Audiobooks API Route
|
* Component: Popular Audiobooks API Route
|
||||||
* Documentation: documentation/integrations/audible.md
|
* Documentation: documentation/integrations/audible.md
|
||||||
*
|
*
|
||||||
* Serves popular audiobooks from audible_cache with real-time Plex matching
|
* Serves popular audiobooks from AudibleCacheCategory with real-time library matching.
|
||||||
|
* Popular books are stored with categoryId '__popular__' in the unified category table.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
import { NextRequest, NextResponse } from 'next/server';
|
import { NextRequest, NextResponse } from 'next/server';
|
||||||
import { prisma } from '@/lib/db';
|
import { prisma } from '@/lib/db';
|
||||||
import { enrichAudiobooksWithMatches } from '@/lib/utils/audiobook-matcher';
|
import { enrichAudiobooksWithMatches, getAvailableAsins } from '@/lib/utils/audiobook-matcher';
|
||||||
import { getCurrentUser } from '@/lib/middleware/auth';
|
import { getCurrentUser } from '@/lib/middleware/auth';
|
||||||
import { RMABLogger } from '@/lib/utils/logger';
|
import { RMABLogger } from '@/lib/utils/logger';
|
||||||
|
import { POPULAR_CATEGORY_ID } from '@/lib/processors/audible-refresh.processor';
|
||||||
|
|
||||||
const logger = RMABLogger.create('API.Audiobooks.Popular');
|
const logger = RMABLogger.create('API.Audiobooks.Popular');
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* GET /api/audiobooks/popular?page=1&limit=20
|
* GET /api/audiobooks/popular?page=1&limit=20
|
||||||
* Get popular audiobooks from audible_cache with pagination
|
* Get popular audiobooks from AudibleCacheCategory with pagination
|
||||||
*
|
*
|
||||||
* Real-time matching against plex_library determines availability.
|
* Real-time matching against plex_library determines availability.
|
||||||
*/
|
*/
|
||||||
@@ -24,6 +26,7 @@ export async function GET(request: NextRequest) {
|
|||||||
const searchParams = request.nextUrl.searchParams;
|
const searchParams = request.nextUrl.searchParams;
|
||||||
const page = parseInt(searchParams.get('page') || '1', 10);
|
const page = parseInt(searchParams.get('page') || '1', 10);
|
||||||
const limit = parseInt(searchParams.get('limit') || '20', 10);
|
const limit = parseInt(searchParams.get('limit') || '20', 10);
|
||||||
|
const hideAvailable = searchParams.get('hideAvailable') === 'true';
|
||||||
|
|
||||||
// Validate pagination parameters
|
// Validate pagination parameters
|
||||||
if (page < 1 || limit < 1 || limit > 100) {
|
if (page < 1 || limit < 1 || limit > 100) {
|
||||||
@@ -38,38 +41,28 @@ export async function GET(request: NextRequest) {
|
|||||||
|
|
||||||
const skip = (page - 1) * limit;
|
const skip = (page - 1) * limit;
|
||||||
|
|
||||||
// Query audible_cache for popular audiobooks
|
// When hideAvailable is enabled, exclude ASINs that are in the library or have completed requests
|
||||||
const [audiobooks, totalCount] = await Promise.all([
|
let excludedAsins: string[] = [];
|
||||||
prisma.audibleCache.findMany({
|
if (hideAvailable) {
|
||||||
where: {
|
const availableSet = await getAvailableAsins();
|
||||||
isPopular: true,
|
excludedAsins = [...availableSet];
|
||||||
},
|
}
|
||||||
orderBy: {
|
|
||||||
popularRank: 'asc',
|
const whereClause: any = { categoryId: POPULAR_CATEGORY_ID };
|
||||||
},
|
if (excludedAsins.length > 0) {
|
||||||
|
whereClause.asin = { notIn: excludedAsins };
|
||||||
|
}
|
||||||
|
|
||||||
|
// Query AudibleCacheCategory for popular audiobooks
|
||||||
|
const [categoryEntries, totalCount] = await Promise.all([
|
||||||
|
prisma.audibleCacheCategory.findMany({
|
||||||
|
where: whereClause,
|
||||||
|
orderBy: { rank: 'asc' },
|
||||||
skip,
|
skip,
|
||||||
take: limit,
|
take: limit,
|
||||||
select: {
|
select: { asin: true, rank: true },
|
||||||
id: true,
|
|
||||||
asin: true,
|
|
||||||
title: true,
|
|
||||||
author: true,
|
|
||||||
narrator: true,
|
|
||||||
description: true,
|
|
||||||
coverArtUrl: true,
|
|
||||||
cachedCoverPath: true,
|
|
||||||
durationMinutes: true,
|
|
||||||
releaseDate: true,
|
|
||||||
rating: true,
|
|
||||||
genres: true,
|
|
||||||
lastSyncedAt: true,
|
|
||||||
},
|
|
||||||
}),
|
|
||||||
prisma.audibleCache.count({
|
|
||||||
where: {
|
|
||||||
isPopular: true,
|
|
||||||
},
|
|
||||||
}),
|
}),
|
||||||
|
prisma.audibleCacheCategory.count({ where: whereClause }),
|
||||||
]);
|
]);
|
||||||
|
|
||||||
// If no data found, return helpful message
|
// If no data found, return helpful message
|
||||||
@@ -86,30 +79,56 @@ export async function GET(request: NextRequest) {
|
|||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
// Transform to matcher input format (uses ASIN as required field)
|
// Fetch full metadata from AudibleCache for these ASINs
|
||||||
// Use cached cover path when available, otherwise fall back to coverArtUrl
|
const asins = categoryEntries.map((e) => e.asin);
|
||||||
const audibleBooks = audiobooks.map((book) => {
|
const cacheEntries = await prisma.audibleCache.findMany({
|
||||||
// Convert cached path to API URL if it exists
|
where: { asin: { in: asins } },
|
||||||
let coverUrl = book.coverArtUrl || undefined;
|
select: {
|
||||||
if (book.cachedCoverPath) {
|
asin: true,
|
||||||
const filename = book.cachedCoverPath.split('/').pop();
|
title: true,
|
||||||
coverUrl = `/api/cache/thumbnails/${filename}`;
|
author: true,
|
||||||
}
|
narrator: true,
|
||||||
|
description: true,
|
||||||
return {
|
coverArtUrl: true,
|
||||||
asin: book.asin,
|
cachedCoverPath: true,
|
||||||
title: book.title,
|
durationMinutes: true,
|
||||||
author: book.author,
|
releaseDate: true,
|
||||||
narrator: book.narrator || undefined,
|
rating: true,
|
||||||
description: book.description || undefined,
|
genres: true,
|
||||||
coverArtUrl: coverUrl,
|
lastSyncedAt: true,
|
||||||
durationMinutes: book.durationMinutes || undefined,
|
},
|
||||||
releaseDate: book.releaseDate?.toISOString() || undefined,
|
|
||||||
rating: book.rating ? parseFloat(book.rating.toString()) : undefined,
|
|
||||||
genres: (book.genres as string[]) || [],
|
|
||||||
};
|
|
||||||
});
|
});
|
||||||
|
|
||||||
|
// Build a map for ordering by rank
|
||||||
|
const cacheMap = new Map(cacheEntries.map((e) => [e.asin, e]));
|
||||||
|
|
||||||
|
// Transform to matcher input format, preserving rank order
|
||||||
|
const audibleBooks = categoryEntries
|
||||||
|
.map((entry) => {
|
||||||
|
const book = cacheMap.get(entry.asin);
|
||||||
|
if (!book) return null;
|
||||||
|
|
||||||
|
let coverUrl = book.coverArtUrl || undefined;
|
||||||
|
if (book.cachedCoverPath) {
|
||||||
|
const filename = book.cachedCoverPath.split('/').pop();
|
||||||
|
coverUrl = `/api/cache/thumbnails/${filename}`;
|
||||||
|
}
|
||||||
|
|
||||||
|
return {
|
||||||
|
asin: book.asin,
|
||||||
|
title: book.title,
|
||||||
|
author: book.author,
|
||||||
|
narrator: book.narrator || undefined,
|
||||||
|
description: book.description || undefined,
|
||||||
|
coverArtUrl: coverUrl,
|
||||||
|
durationMinutes: book.durationMinutes || undefined,
|
||||||
|
releaseDate: book.releaseDate?.toISOString() || undefined,
|
||||||
|
rating: book.rating ? parseFloat(book.rating.toString()) : undefined,
|
||||||
|
genres: (book.genres as string[]) || [],
|
||||||
|
};
|
||||||
|
})
|
||||||
|
.filter(Boolean) as any[];
|
||||||
|
|
||||||
// Get current user (optional - for request status enrichment)
|
// Get current user (optional - for request status enrichment)
|
||||||
const currentUser = getCurrentUser(request);
|
const currentUser = getCurrentUser(request);
|
||||||
const userId = currentUser?.sub || undefined;
|
const userId = currentUser?.sub || undefined;
|
||||||
@@ -128,7 +147,7 @@ export async function GET(request: NextRequest) {
|
|||||||
page,
|
page,
|
||||||
totalPages,
|
totalPages,
|
||||||
hasMore,
|
hasMore,
|
||||||
lastSync: audiobooks[0]?.lastSyncedAt?.toISOString() || null,
|
lastSync: cacheEntries[0]?.lastSyncedAt?.toISOString() || null,
|
||||||
});
|
});
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
logger.error('Failed to get popular audiobooks', { error: error instanceof Error ? error.message : String(error) });
|
logger.error('Failed to get popular audiobooks', { error: error instanceof Error ? error.message : String(error) });
|
||||||
|
|||||||
@@ -6,6 +6,8 @@
|
|||||||
import { NextRequest, NextResponse } from 'next/server';
|
import { NextRequest, NextResponse } from 'next/server';
|
||||||
import { getAudibleService } from '@/lib/integrations/audible.service';
|
import { getAudibleService } from '@/lib/integrations/audible.service';
|
||||||
import { enrichAudiobooksWithMatches } from '@/lib/utils/audiobook-matcher';
|
import { enrichAudiobooksWithMatches } from '@/lib/utils/audiobook-matcher';
|
||||||
|
import { deduplicateAndCollectGroups } from '@/lib/utils/deduplicate-audiobooks';
|
||||||
|
import { persistDedupGroups } from '@/lib/services/works.service';
|
||||||
import { getCurrentUser } from '@/lib/middleware/auth';
|
import { getCurrentUser } from '@/lib/middleware/auth';
|
||||||
import { RMABLogger } from '@/lib/utils/logger';
|
import { RMABLogger } from '@/lib/utils/logger';
|
||||||
|
|
||||||
@@ -38,14 +40,22 @@ export async function GET(request: NextRequest) {
|
|||||||
const currentUser = getCurrentUser(request);
|
const currentUser = getCurrentUser(request);
|
||||||
const userId = currentUser?.sub || undefined;
|
const userId = currentUser?.sub || undefined;
|
||||||
|
|
||||||
|
// Deduplicate before enrichment to avoid wasted DB queries on duplicate entries
|
||||||
|
const { books: dedupedResults, groups } = deduplicateAndCollectGroups(results.results);
|
||||||
|
|
||||||
|
// Fire-and-forget: persist dedup groups to works table for cross-ASIN matching
|
||||||
|
if (groups.length > 0) {
|
||||||
|
persistDedupGroups(groups).catch(() => {});
|
||||||
|
}
|
||||||
|
|
||||||
// Enrich search results with availability and request status information
|
// Enrich search results with availability and request status information
|
||||||
const enrichedResults = await enrichAudiobooksWithMatches(results.results, userId);
|
const enrichedResults = await enrichAudiobooksWithMatches(dedupedResults, userId);
|
||||||
|
|
||||||
return NextResponse.json({
|
return NextResponse.json({
|
||||||
success: true,
|
success: true,
|
||||||
query: results.query,
|
query: results.query,
|
||||||
results: enrichedResults,
|
results: enrichedResults,
|
||||||
totalResults: results.totalResults,
|
totalResults: enrichedResults.length,
|
||||||
page: results.page,
|
page: results.page,
|
||||||
hasMore: results.hasMore,
|
hasMore: results.hasMore,
|
||||||
});
|
});
|
||||||
|
|||||||
@@ -38,9 +38,11 @@ export async function POST(request: NextRequest) {
|
|||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
const normalizedUsername = username.trim().toLowerCase();
|
||||||
|
|
||||||
// Find user by local admin identifier
|
// Find user by local admin identifier
|
||||||
const user = await prisma.user.findUnique({
|
const user = await prisma.user.findUnique({
|
||||||
where: { plexId: `local-${username}` },
|
where: { plexId: `local-${normalizedUsername}` },
|
||||||
});
|
});
|
||||||
|
|
||||||
if (!user) {
|
if (!user) {
|
||||||
|
|||||||
@@ -39,6 +39,7 @@ export async function GET(request: NextRequest) {
|
|||||||
createdAt: true,
|
createdAt: true,
|
||||||
lastLoginAt: true,
|
lastLoginAt: true,
|
||||||
interactiveSearchAccess: true,
|
interactiveSearchAccess: true,
|
||||||
|
downloadAccess: true,
|
||||||
},
|
},
|
||||||
});
|
});
|
||||||
|
|
||||||
@@ -63,6 +64,13 @@ export async function GET(request: NextRequest) {
|
|||||||
globalInteractiveSearch
|
globalInteractiveSearch
|
||||||
);
|
);
|
||||||
|
|
||||||
|
const globalDownload = await getGlobalBooleanSetting('download_access', true);
|
||||||
|
const effectiveDownload = resolvePermission(
|
||||||
|
user.role,
|
||||||
|
user.downloadAccess,
|
||||||
|
globalDownload
|
||||||
|
);
|
||||||
|
|
||||||
return NextResponse.json({
|
return NextResponse.json({
|
||||||
user: {
|
user: {
|
||||||
id: user.id,
|
id: user.id,
|
||||||
@@ -77,6 +85,7 @@ export async function GET(request: NextRequest) {
|
|||||||
lastLoginAt: user.lastLoginAt,
|
lastLoginAt: user.lastLoginAt,
|
||||||
permissions: {
|
permissions: {
|
||||||
interactiveSearch: effectiveInteractiveSearch,
|
interactiveSearch: effectiveInteractiveSearch,
|
||||||
|
download: effectiveDownload,
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
});
|
});
|
||||||
|
|||||||
@@ -6,6 +6,8 @@
|
|||||||
import { NextRequest, NextResponse } from 'next/server';
|
import { NextRequest, NextResponse } from 'next/server';
|
||||||
import { getAudibleService } from '@/lib/integrations/audible.service';
|
import { getAudibleService } from '@/lib/integrations/audible.service';
|
||||||
import { enrichAudiobooksWithMatches } from '@/lib/utils/audiobook-matcher';
|
import { enrichAudiobooksWithMatches } from '@/lib/utils/audiobook-matcher';
|
||||||
|
import { deduplicateAndCollectGroups } from '@/lib/utils/deduplicate-audiobooks';
|
||||||
|
import { persistDedupGroups } from '@/lib/services/works.service';
|
||||||
import { getCurrentUser } from '@/lib/middleware/auth';
|
import { getCurrentUser } from '@/lib/middleware/auth';
|
||||||
import { RMABLogger } from '@/lib/utils/logger';
|
import { RMABLogger } from '@/lib/utils/logger';
|
||||||
|
|
||||||
@@ -46,16 +48,26 @@ export async function GET(
|
|||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
logger.info(`Fetching books for author "${authorName}" (ASIN: ${asin})`);
|
const page = parseInt(request.nextUrl.searchParams.get('page') || '1', 10);
|
||||||
|
|
||||||
|
logger.info(`Fetching books for author "${authorName}" (ASIN: ${asin}), page ${page}`);
|
||||||
|
|
||||||
const audibleService = getAudibleService();
|
const audibleService = getAudibleService();
|
||||||
const books = await audibleService.searchByAuthorAsin(authorName.trim(), asin);
|
const result = await audibleService.searchByAuthorAsin(authorName.trim(), asin, page);
|
||||||
|
|
||||||
|
// Deduplicate before enrichment to avoid wasted DB queries on duplicate entries
|
||||||
|
const { books: dedupedBooks, groups } = deduplicateAndCollectGroups(result.books);
|
||||||
|
|
||||||
|
// Fire-and-forget: persist dedup groups to works table for cross-ASIN matching
|
||||||
|
if (groups.length > 0) {
|
||||||
|
persistDedupGroups(groups).catch(() => {});
|
||||||
|
}
|
||||||
|
|
||||||
// Enrich with library availability and request status
|
// Enrich with library availability and request status
|
||||||
const userId = currentUser.sub || undefined;
|
const userId = currentUser.sub || undefined;
|
||||||
const enrichedBooks = await enrichAudiobooksWithMatches(books, userId);
|
const enrichedBooks = await enrichAudiobooksWithMatches(dedupedBooks, userId);
|
||||||
|
|
||||||
logger.info(`Author books complete: "${authorName}" → ${enrichedBooks.length} books`);
|
logger.info(`Author books complete: "${authorName}" → ${enrichedBooks.length} books (page ${page})`);
|
||||||
|
|
||||||
return NextResponse.json({
|
return NextResponse.json({
|
||||||
success: true,
|
success: true,
|
||||||
@@ -63,6 +75,8 @@ export async function GET(
|
|||||||
authorName: authorName.trim(),
|
authorName: authorName.trim(),
|
||||||
authorAsin: asin,
|
authorAsin: asin,
|
||||||
totalBooks: enrichedBooks.length,
|
totalBooks: enrichedBooks.length,
|
||||||
|
hasMore: result.hasMore,
|
||||||
|
page: result.page,
|
||||||
});
|
});
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
logger.error('Failed to fetch author books', { error: error instanceof Error ? error.message : String(error) });
|
logger.error('Failed to fetch author books', { error: error instanceof Error ? error.message : String(error) });
|
||||||
|
|||||||
@@ -59,9 +59,9 @@ async function saveConfig(req: AuthenticatedRequest) {
|
|||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!['openai', 'claude', 'custom'].includes(provider)) {
|
if (!['openai', 'claude', 'custom', 'gemini'].includes(provider)) {
|
||||||
return NextResponse.json(
|
return NextResponse.json(
|
||||||
{ error: 'Invalid provider. Must be "openai", "claude", or "custom"' },
|
{ error: 'Invalid provider. Must be "openai", "claude", "custom", or "gemini"' },
|
||||||
{ status: 400 }
|
{ status: 400 }
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
@@ -107,7 +107,7 @@ async function saveConfig(req: AuthenticatedRequest) {
|
|||||||
// No new API key, use existing one
|
// No new API key, use existing one
|
||||||
encryptedApiKeyToUse = existingConfig.apiKey;
|
encryptedApiKeyToUse = existingConfig.apiKey;
|
||||||
} else {
|
} else {
|
||||||
// API key required for OpenAI/Claude
|
// API key required for OpenAI/Claude/Gemini
|
||||||
return NextResponse.json(
|
return NextResponse.json(
|
||||||
{ error: 'API key is required' },
|
{ error: 'API key is required' },
|
||||||
{ status: 400 }
|
{ status: 400 }
|
||||||
|
|||||||
@@ -52,6 +52,30 @@ async function fetchClaudeModels(apiKey: string): Promise<{ id: string; name: st
|
|||||||
return allModels;
|
return allModels;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Fetch available Gemini models from the Google API
|
||||||
|
async function fetchGeminiModels(apiKey: string): Promise<{ id: string; name: string }[]> {
|
||||||
|
const response = await fetch(
|
||||||
|
'https://generativelanguage.googleapis.com/v1beta/models',
|
||||||
|
{ headers: { 'x-goog-api-key': apiKey } }
|
||||||
|
);
|
||||||
|
|
||||||
|
if (!response.ok) {
|
||||||
|
const errorText = await response.text();
|
||||||
|
logger.error('Gemini API error', { error: errorText });
|
||||||
|
throw new Error('Invalid Gemini API key or connection failed');
|
||||||
|
}
|
||||||
|
|
||||||
|
const data = await response.json();
|
||||||
|
|
||||||
|
return (data.models || [])
|
||||||
|
.filter((m: any) => m.name?.startsWith('models/gemini-') && m.supportedGenerationMethods?.includes('generateContent'))
|
||||||
|
.map((m: any) => ({
|
||||||
|
id: m.name.replace('models/', ''),
|
||||||
|
name: m.displayName || m.name.replace('models/', ''),
|
||||||
|
}))
|
||||||
|
.sort((a: any, b: any) => a.name.localeCompare(b.name));
|
||||||
|
}
|
||||||
|
|
||||||
// Helper functions for custom provider
|
// Helper functions for custom provider
|
||||||
function isValidBaseUrl(url: string): boolean {
|
function isValidBaseUrl(url: string): boolean {
|
||||||
try {
|
try {
|
||||||
@@ -79,9 +103,9 @@ async function authenticatedHandler(req: AuthenticatedRequest) {
|
|||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!['openai', 'claude', 'custom'].includes(provider)) {
|
if (!['openai', 'claude', 'custom', 'gemini'].includes(provider)) {
|
||||||
return NextResponse.json(
|
return NextResponse.json(
|
||||||
{ error: 'Invalid provider. Must be "openai", "claude", or "custom"' },
|
{ error: 'Invalid provider. Must be "openai", "claude", "custom", or "gemini"' },
|
||||||
{ status: 400 }
|
{ status: 400 }
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
@@ -193,6 +217,16 @@ async function authenticatedHandler(req: AuthenticatedRequest) {
|
|||||||
{ status: 400 }
|
{ status: 400 }
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
} else if (provider === 'gemini') {
|
||||||
|
// Gemini: Fetch models dynamically from the Google API
|
||||||
|
try {
|
||||||
|
models = await fetchGeminiModels(testApiKey);
|
||||||
|
} catch {
|
||||||
|
return NextResponse.json(
|
||||||
|
{ error: 'Invalid Gemini API key or connection failed' },
|
||||||
|
{ status: 400 }
|
||||||
|
);
|
||||||
|
}
|
||||||
} else if (provider === 'custom') {
|
} else if (provider === 'custom') {
|
||||||
// Custom: Fetch models from custom OpenAI-compatible endpoint
|
// Custom: Fetch models from custom OpenAI-compatible endpoint
|
||||||
const normalizedUrl = normalizeBaseUrl(testBaseUrl);
|
const normalizedUrl = normalizeBaseUrl(testBaseUrl);
|
||||||
@@ -291,9 +325,9 @@ async function unauthenticatedHandler(req: NextRequest) {
|
|||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!['openai', 'claude', 'custom'].includes(provider)) {
|
if (!['openai', 'claude', 'custom', 'gemini'].includes(provider)) {
|
||||||
return NextResponse.json(
|
return NextResponse.json(
|
||||||
{ error: 'Invalid provider. Must be "openai", "claude", or "custom"' },
|
{ error: 'Invalid provider. Must be "openai", "claude", "custom", or "gemini"' },
|
||||||
{ status: 400 }
|
{ status: 400 }
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
@@ -363,6 +397,16 @@ async function unauthenticatedHandler(req: NextRequest) {
|
|||||||
{ status: 400 }
|
{ status: 400 }
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
} else if (provider === 'gemini') {
|
||||||
|
// Gemini: Fetch models dynamically
|
||||||
|
try {
|
||||||
|
models = await fetchGeminiModels(apiKey);
|
||||||
|
} catch {
|
||||||
|
return NextResponse.json(
|
||||||
|
{ error: 'Invalid Gemini API key or connection failed' },
|
||||||
|
{ status: 400 }
|
||||||
|
);
|
||||||
|
}
|
||||||
} else if (provider === 'custom') {
|
} else if (provider === 'custom') {
|
||||||
// Custom: Fetch models from custom OpenAI-compatible endpoint
|
// Custom: Fetch models from custom OpenAI-compatible endpoint
|
||||||
const normalizedUrl = normalizeBaseUrl(baseUrl);
|
const normalizedUrl = normalizeBaseUrl(baseUrl);
|
||||||
|
|||||||
@@ -0,0 +1,89 @@
|
|||||||
|
/**
|
||||||
|
* Component: On-Demand Download Token Generator
|
||||||
|
* Documentation: documentation/backend/api.md
|
||||||
|
*/
|
||||||
|
|
||||||
|
import { NextRequest, NextResponse } from 'next/server';
|
||||||
|
import { requireAuth, AuthenticatedRequest } from '@/lib/middleware/auth';
|
||||||
|
import { prisma } from '@/lib/db';
|
||||||
|
import { generateDownloadToken } from '@/lib/utils/jwt';
|
||||||
|
import { COMPLETED_STATUSES } from '@/lib/constants/request-statuses';
|
||||||
|
import { resolveDownloadAccess } from '@/lib/utils/permissions';
|
||||||
|
import { RMABLogger } from '@/lib/utils/logger';
|
||||||
|
|
||||||
|
const logger = RMABLogger.create('API.DownloadToken');
|
||||||
|
|
||||||
|
/**
|
||||||
|
* POST /api/requests/[id]/download-token
|
||||||
|
* Generate a signed download token on demand (lazy token generation).
|
||||||
|
*/
|
||||||
|
export async function POST(
|
||||||
|
request: NextRequest,
|
||||||
|
{ params }: { params: Promise<{ id: string }> }
|
||||||
|
) {
|
||||||
|
return requireAuth(request, async (req: AuthenticatedRequest) => {
|
||||||
|
try {
|
||||||
|
if (!req.user) {
|
||||||
|
return NextResponse.json(
|
||||||
|
{ error: 'Unauthorized', message: 'User not authenticated' },
|
||||||
|
{ status: 401 }
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Check download permission
|
||||||
|
const userRecord = await prisma.user.findUnique({
|
||||||
|
where: { id: req.user.id },
|
||||||
|
select: { role: true, downloadAccess: true },
|
||||||
|
});
|
||||||
|
const hasDownloadAccess = await resolveDownloadAccess(
|
||||||
|
userRecord?.role ?? 'user',
|
||||||
|
userRecord?.downloadAccess ?? null
|
||||||
|
);
|
||||||
|
if (!hasDownloadAccess) {
|
||||||
|
return NextResponse.json(
|
||||||
|
{ error: 'Forbidden', message: 'You do not have download access' },
|
||||||
|
{ status: 403 }
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
const { id } = await params;
|
||||||
|
|
||||||
|
const requestRecord = await prisma.request.findFirst({
|
||||||
|
where: { id, deletedAt: null },
|
||||||
|
include: { audiobook: true },
|
||||||
|
});
|
||||||
|
|
||||||
|
if (!requestRecord) {
|
||||||
|
return NextResponse.json(
|
||||||
|
{ error: 'NotFound', message: 'Request not found' },
|
||||||
|
{ status: 404 }
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!COMPLETED_STATUSES.includes(requestRecord.status as typeof COMPLETED_STATUSES[number])) {
|
||||||
|
return NextResponse.json(
|
||||||
|
{ error: 'BadRequest', message: 'Request is not yet completed' },
|
||||||
|
{ status: 400 }
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!requestRecord.audiobook?.filePath) {
|
||||||
|
return NextResponse.json(
|
||||||
|
{ error: 'NotFound', message: 'No file available for this request' },
|
||||||
|
{ status: 404 }
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
const token = generateDownloadToken(req.user.id, id);
|
||||||
|
const downloadUrl = `/api/requests/${id}/download?token=${token}`;
|
||||||
|
|
||||||
|
return NextResponse.json({ downloadUrl });
|
||||||
|
} catch (error) {
|
||||||
|
logger.error('Failed to generate download token', { error: error instanceof Error ? error.message : String(error) });
|
||||||
|
return NextResponse.json(
|
||||||
|
{ error: 'TokenError', message: 'Failed to generate download token' },
|
||||||
|
{ status: 500 }
|
||||||
|
);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
@@ -0,0 +1,152 @@
|
|||||||
|
/**
|
||||||
|
* Component: Request File Download Endpoint
|
||||||
|
* Documentation: documentation/backend/api.md
|
||||||
|
*/
|
||||||
|
|
||||||
|
import { NextRequest, NextResponse } from 'next/server';
|
||||||
|
import { prisma } from '@/lib/db';
|
||||||
|
import { verifyDownloadToken } from '@/lib/utils/jwt';
|
||||||
|
import { RMABLogger } from '@/lib/utils/logger';
|
||||||
|
import { AUDIO_EXTENSIONS, EBOOK_EXTENSIONS } from '@/lib/constants/audio-formats';
|
||||||
|
import { COMPLETED_STATUSES } from '@/lib/constants/request-statuses';
|
||||||
|
import fs from 'fs';
|
||||||
|
import path from 'path';
|
||||||
|
import archiver from 'archiver';
|
||||||
|
import { PassThrough } from 'stream';
|
||||||
|
|
||||||
|
const logger = RMABLogger.create('API.Download');
|
||||||
|
|
||||||
|
function sanitizeFilename(name: string): string {
|
||||||
|
return name
|
||||||
|
.replace(/[<>:"/\\|?*]/g, '')
|
||||||
|
.replace(/\s+/g, ' ')
|
||||||
|
.trim()
|
||||||
|
.slice(0, 200);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* GET /api/requests/[id]/download?token=<JWT>
|
||||||
|
* Token-authenticated file download — no session cookie required.
|
||||||
|
*/
|
||||||
|
export async function GET(
|
||||||
|
request: NextRequest,
|
||||||
|
{ params }: { params: Promise<{ id: string }> }
|
||||||
|
) {
|
||||||
|
try {
|
||||||
|
const { id } = await params;
|
||||||
|
const token = request.nextUrl.searchParams.get('token');
|
||||||
|
|
||||||
|
if (!token) {
|
||||||
|
return NextResponse.json({ error: 'Unauthorized', message: 'Missing download token' }, { status: 401 });
|
||||||
|
}
|
||||||
|
|
||||||
|
const payload = verifyDownloadToken(token);
|
||||||
|
if (!payload) {
|
||||||
|
return NextResponse.json({ error: 'Unauthorized', message: 'Invalid or expired download token' }, { status: 401 });
|
||||||
|
}
|
||||||
|
|
||||||
|
if (payload.requestId !== id) {
|
||||||
|
return NextResponse.json({ error: 'Unauthorized', message: 'Token does not match request' }, { status: 401 });
|
||||||
|
}
|
||||||
|
|
||||||
|
const requestRecord = await prisma.request.findFirst({
|
||||||
|
where: { id, deletedAt: null },
|
||||||
|
include: { audiobook: true },
|
||||||
|
});
|
||||||
|
|
||||||
|
if (!requestRecord) {
|
||||||
|
return NextResponse.json({ error: 'NotFound', message: 'Request not found' }, { status: 404 });
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!COMPLETED_STATUSES.includes(requestRecord.status as typeof COMPLETED_STATUSES[number])) {
|
||||||
|
return NextResponse.json({ error: 'BadRequest', message: 'Request is not yet completed' }, { status: 400 });
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!requestRecord.audiobook?.filePath) {
|
||||||
|
return NextResponse.json({ error: 'NotFound', message: 'No file path available for this request' }, { status: 404 });
|
||||||
|
}
|
||||||
|
|
||||||
|
const resolvedDir = path.resolve(requestRecord.audiobook.filePath);
|
||||||
|
|
||||||
|
if (!fs.existsSync(resolvedDir)) {
|
||||||
|
logger.error('Download directory does not exist', { path: resolvedDir });
|
||||||
|
return NextResponse.json({ error: 'NotFound', message: 'File directory not found on disk' }, { status: 404 });
|
||||||
|
}
|
||||||
|
|
||||||
|
const requestType = requestRecord.type || 'audiobook';
|
||||||
|
const allowedExtensions: readonly string[] = requestType === 'ebook' ? EBOOK_EXTENSIONS : AUDIO_EXTENSIONS;
|
||||||
|
|
||||||
|
const allEntries = fs.readdirSync(resolvedDir);
|
||||||
|
const matchingFiles = allEntries
|
||||||
|
.filter(name => allowedExtensions.includes(path.extname(name).toLowerCase()))
|
||||||
|
.map(name => path.join(resolvedDir, name));
|
||||||
|
|
||||||
|
if (matchingFiles.length === 0) {
|
||||||
|
return NextResponse.json({ error: 'NotFound', message: 'No matching files found in directory' }, { status: 404 });
|
||||||
|
}
|
||||||
|
|
||||||
|
const sanitizedTitle = sanitizeFilename(requestRecord.audiobook.title || 'download');
|
||||||
|
|
||||||
|
if (matchingFiles.length === 1) {
|
||||||
|
const filePath = matchingFiles[0];
|
||||||
|
const ext = path.extname(filePath);
|
||||||
|
const stat = fs.statSync(filePath);
|
||||||
|
const fileStream = fs.createReadStream(filePath);
|
||||||
|
|
||||||
|
const readableStream = new ReadableStream({
|
||||||
|
start(controller) {
|
||||||
|
fileStream.on('data', chunk => controller.enqueue(chunk));
|
||||||
|
fileStream.on('end', () => controller.close());
|
||||||
|
fileStream.on('error', err => {
|
||||||
|
logger.error('File stream error', { error: err.message });
|
||||||
|
controller.error(err);
|
||||||
|
});
|
||||||
|
},
|
||||||
|
cancel() {
|
||||||
|
fileStream.destroy();
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
return new NextResponse(readableStream, {
|
||||||
|
headers: {
|
||||||
|
'Content-Type': 'application/octet-stream',
|
||||||
|
'Content-Disposition': `attachment; filename="${sanitizedTitle}${ext}"`,
|
||||||
|
'Content-Length': String(stat.size),
|
||||||
|
},
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
// Multiple files — stream zip via archiver (avoids loading all files into memory)
|
||||||
|
const passThrough = new PassThrough();
|
||||||
|
const archive = archiver('zip', { zlib: { level: 6 } });
|
||||||
|
archive.pipe(passThrough);
|
||||||
|
for (const filePath of matchingFiles) {
|
||||||
|
archive.file(filePath, { name: path.basename(filePath) });
|
||||||
|
}
|
||||||
|
archive.finalize();
|
||||||
|
|
||||||
|
const zipReadable = new ReadableStream({
|
||||||
|
start(controller) {
|
||||||
|
passThrough.on('data', chunk => controller.enqueue(new Uint8Array(chunk)));
|
||||||
|
passThrough.on('end', () => controller.close());
|
||||||
|
passThrough.on('error', err => {
|
||||||
|
logger.error('Zip stream error', { error: err.message });
|
||||||
|
controller.error(err);
|
||||||
|
});
|
||||||
|
},
|
||||||
|
cancel() {
|
||||||
|
archive.abort();
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
return new NextResponse(zipReadable, {
|
||||||
|
headers: {
|
||||||
|
'Content-Type': 'application/zip',
|
||||||
|
'Content-Disposition': `attachment; filename="${sanitizedTitle}.zip"`,
|
||||||
|
},
|
||||||
|
});
|
||||||
|
} catch (error) {
|
||||||
|
logger.error('Download failed', { error: error instanceof Error ? error.message : String(error) });
|
||||||
|
return NextResponse.json({ error: 'DownloadError', message: 'Failed to serve file' }, { status: 500 });
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -71,41 +71,56 @@ export async function POST(
|
|||||||
const body = await request.json().catch(() => ({}));
|
const body = await request.json().catch(() => ({}));
|
||||||
const customTitle = body.customTitle as string | undefined;
|
const customTitle = body.customTitle as string | undefined;
|
||||||
|
|
||||||
// Get the parent audiobook request
|
// Get the request (can be audiobook parent or direct ebook request)
|
||||||
const parentRequest = await prisma.request.findUnique({
|
const requestRecord = await prisma.request.findUnique({
|
||||||
where: { id: parentRequestId },
|
where: { id: parentRequestId },
|
||||||
include: { audiobook: true },
|
include: { audiobook: true },
|
||||||
});
|
});
|
||||||
|
|
||||||
if (!parentRequest) {
|
if (!requestRecord) {
|
||||||
return NextResponse.json({ error: 'Request not found' }, { status: 404 });
|
return NextResponse.json({ error: 'Request not found' }, { status: 404 });
|
||||||
}
|
}
|
||||||
|
|
||||||
if (parentRequest.type !== 'audiobook') {
|
// Support two flows:
|
||||||
return NextResponse.json({ error: 'Can only search ebooks for audiobook requests' }, { status: 400 });
|
// Flow A (sidecar): Audiobook request in downloaded/available state
|
||||||
|
// Flow B (direct): Ebook request in pending/failed/awaiting_search state
|
||||||
|
const isDirectEbookSearch = requestRecord.type === 'ebook';
|
||||||
|
const isAudiobookSidecar = requestRecord.type === 'audiobook';
|
||||||
|
|
||||||
|
if (!isDirectEbookSearch && !isAudiobookSidecar) {
|
||||||
|
return NextResponse.json({ error: 'Invalid request type' }, { status: 400 });
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!['downloaded', 'available'].includes(parentRequest.status)) {
|
if (isAudiobookSidecar && !['downloaded', 'available'].includes(requestRecord.status)) {
|
||||||
return NextResponse.json(
|
return NextResponse.json(
|
||||||
{ error: `Cannot search ebooks for request in ${parentRequest.status} status` },
|
{ error: `Cannot search ebooks for audiobook request in ${requestRecord.status} status` },
|
||||||
{ status: 400 }
|
{ status: 400 }
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
// Check for existing non-retryable ebook request
|
if (isDirectEbookSearch && !['pending', 'failed', 'awaiting_search'].includes(requestRecord.status)) {
|
||||||
const existingEbookRequest = await prisma.request.findFirst({
|
return NextResponse.json(
|
||||||
where: {
|
{ error: `Cannot search for ebook request in ${requestRecord.status} status` },
|
||||||
parentRequestId,
|
{ status: 400 }
|
||||||
type: 'ebook',
|
);
|
||||||
deletedAt: null,
|
}
|
||||||
},
|
|
||||||
});
|
|
||||||
|
|
||||||
if (existingEbookRequest && !['failed', 'awaiting_search'].includes(existingEbookRequest.status)) {
|
// Check for existing child ebook requests (sidecar mode only)
|
||||||
return NextResponse.json({
|
if (isAudiobookSidecar) {
|
||||||
error: `E-book request already exists (status: ${existingEbookRequest.status})`,
|
const existingEbookRequest = await prisma.request.findFirst({
|
||||||
existingRequestId: existingEbookRequest.id,
|
where: {
|
||||||
}, { status: 400 });
|
parentRequestId,
|
||||||
|
type: 'ebook',
|
||||||
|
deletedAt: null,
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
if (existingEbookRequest && !['failed', 'awaiting_search'].includes(existingEbookRequest.status)) {
|
||||||
|
return NextResponse.json({
|
||||||
|
error: `E-book request already exists (status: ${existingEbookRequest.status})`,
|
||||||
|
existingRequestId: existingEbookRequest.id,
|
||||||
|
}, { status: 400 });
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Get ebook configuration
|
// Get ebook configuration
|
||||||
@@ -121,7 +136,7 @@ export async function POST(
|
|||||||
const isAnnasArchiveEnabled = annasArchiveEnabled === 'true';
|
const isAnnasArchiveEnabled = annasArchiveEnabled === 'true';
|
||||||
const isIndexerSearchEnabled = indexerSearchEnabled === 'true';
|
const isIndexerSearchEnabled = indexerSearchEnabled === 'true';
|
||||||
const format = preferredFormat || 'epub';
|
const format = preferredFormat || 'epub';
|
||||||
const annasBaseUrl = baseUrl || 'https://annas-archive.li';
|
const annasBaseUrl = baseUrl || 'https://annas-archive.gl';
|
||||||
|
|
||||||
// Get language code from Audible region config
|
// Get language code from Audible region config
|
||||||
const region = await configService.getAudibleRegion() as AudibleRegion;
|
const region = await configService.getAudibleRegion() as AudibleRegion;
|
||||||
@@ -135,10 +150,10 @@ export async function POST(
|
|||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
const audiobook = parentRequest.audiobook;
|
const audiobook = requestRecord.audiobook;
|
||||||
const searchTitle = customTitle || audiobook.title;
|
const searchTitle = customTitle || audiobook.title;
|
||||||
|
|
||||||
logger.info(`Interactive ebook search for "${searchTitle}" by ${audiobook.author}`);
|
logger.info(`Interactive ebook search for "${searchTitle}" by ${audiobook.author} (${isDirectEbookSearch ? 'direct' : 'sidecar'})`);
|
||||||
logger.info(`Sources: Anna's Archive=${isAnnasArchiveEnabled}, Indexer=${isIndexerSearchEnabled}`);
|
logger.info(`Sources: Anna's Archive=${isAnnasArchiveEnabled}, Indexer=${isIndexerSearchEnabled}`);
|
||||||
|
|
||||||
// Search both sources in parallel
|
// Search both sources in parallel
|
||||||
|
|||||||
@@ -67,8 +67,8 @@ export async function POST(
|
|||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
// Check if request is awaiting approval
|
// Check if request is awaiting approval (admins can still search to override the user's selection)
|
||||||
if (requestRecord.status === 'awaiting_approval') {
|
if (requestRecord.status === 'awaiting_approval' && req.user.role !== 'admin') {
|
||||||
return NextResponse.json(
|
return NextResponse.json(
|
||||||
{ error: 'AwaitingApproval', message: 'This request is awaiting admin approval. You cannot search for torrents until it is approved.' },
|
{ error: 'AwaitingApproval', message: 'This request is awaiting admin approval. You cannot search for torrents until it is approved.' },
|
||||||
{ status: 403 }
|
{ status: 403 }
|
||||||
@@ -125,8 +125,8 @@ export async function POST(
|
|||||||
logger.info(`Skipping ${skippedIndexers.length} indexer(s) with no audiobook categories: ${skippedNames}`);
|
logger.info(`Skipping ${skippedIndexers.length} indexer(s) with no audiobook categories: ${skippedNames}`);
|
||||||
}
|
}
|
||||||
|
|
||||||
// Use custom title if provided, otherwise use audiobook's title
|
// Use custom title if provided, then custom search terms, then audiobook's title
|
||||||
const searchTitle = customTitle || requestRecord.audiobook.title;
|
const searchTitle = customTitle || requestRecord.customSearchTerms || requestRecord.audiobook.title;
|
||||||
const searchAuthor = requestRecord.audiobook.author;
|
const searchAuthor = requestRecord.audiobook.author;
|
||||||
|
|
||||||
logger.info(`Searching ${indexersConfig.length - skippedIndexers.length} enabled indexers in ${groups.length} group${groups.length > 1 ? 's' : ''}`, { searchTitle });
|
logger.info(`Searching ${indexersConfig.length - skippedIndexers.length} enabled indexers in ${groups.length} group${groups.length > 1 ? 's' : ''}`, { searchTitle });
|
||||||
|
|||||||
@@ -64,14 +64,20 @@ export async function POST(
|
|||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
// Trigger search job
|
// Trigger appropriate search job based on request type
|
||||||
const jobQueue = getJobQueueService();
|
const jobQueue = getJobQueueService();
|
||||||
await jobQueue.addSearchJob(id, {
|
const audiobookData = {
|
||||||
id: requestRecord.audiobook.id,
|
id: requestRecord.audiobook.id,
|
||||||
title: requestRecord.audiobook.title,
|
title: requestRecord.audiobook.title,
|
||||||
author: requestRecord.audiobook.author,
|
author: requestRecord.audiobook.author,
|
||||||
asin: requestRecord.audiobook.audibleAsin || undefined,
|
asin: requestRecord.audiobook.audibleAsin || undefined,
|
||||||
});
|
};
|
||||||
|
|
||||||
|
if (requestRecord.type === 'ebook') {
|
||||||
|
await jobQueue.addSearchEbookJob(id, audiobookData);
|
||||||
|
} else {
|
||||||
|
await jobQueue.addSearchJob(id, audiobookData);
|
||||||
|
}
|
||||||
|
|
||||||
// Update request status
|
// Update request status
|
||||||
const updated = await prisma.request.update({
|
const updated = await prisma.request.update({
|
||||||
|
|||||||
@@ -9,6 +9,7 @@ import { prisma } from '@/lib/db';
|
|||||||
import { z } from 'zod';
|
import { z } from 'zod';
|
||||||
import { RMABLogger } from '@/lib/utils/logger';
|
import { RMABLogger } from '@/lib/utils/logger';
|
||||||
import { createRequestForUser } from '@/lib/services/request-creator.service';
|
import { createRequestForUser } from '@/lib/services/request-creator.service';
|
||||||
|
import { COMPLETED_STATUSES } from '@/lib/constants/request-statuses';
|
||||||
|
|
||||||
const logger = RMABLogger.create('API.Requests');
|
const logger = RMABLogger.create('API.Requests');
|
||||||
|
|
||||||
@@ -146,10 +147,18 @@ export async function GET(request: NextRequest) {
|
|||||||
take: limit,
|
take: limit,
|
||||||
});
|
});
|
||||||
|
|
||||||
|
const enriched = requests.map(r => {
|
||||||
|
const isCompleted = COMPLETED_STATUSES.includes(r.status as typeof COMPLETED_STATUSES[number]);
|
||||||
|
const downloadAvailable = isCompleted && !!r.audiobook?.filePath;
|
||||||
|
// Strip server-side absolute path from client response
|
||||||
|
const audiobook = r.audiobook ? { ...r.audiobook, filePath: undefined } : r.audiobook;
|
||||||
|
return { ...r, audiobook, downloadAvailable };
|
||||||
|
});
|
||||||
|
|
||||||
return NextResponse.json({
|
return NextResponse.json({
|
||||||
success: true,
|
success: true,
|
||||||
requests,
|
requests: enriched,
|
||||||
count: requests.length,
|
count: enriched.length,
|
||||||
});
|
});
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
logger.error('Failed to get requests', { error: error instanceof Error ? error.message : String(error) });
|
logger.error('Failed to get requests', { error: error instanceof Error ? error.message : String(error) });
|
||||||
|
|||||||
@@ -8,6 +8,8 @@ import { getCurrentUser } from '@/lib/middleware/auth';
|
|||||||
import { RMABLogger } from '@/lib/utils/logger';
|
import { RMABLogger } from '@/lib/utils/logger';
|
||||||
import { scrapeSeriesPage } from '@/lib/integrations/audible-series';
|
import { scrapeSeriesPage } from '@/lib/integrations/audible-series';
|
||||||
import { enrichAudiobooksWithMatches } from '@/lib/utils/audiobook-matcher';
|
import { enrichAudiobooksWithMatches } from '@/lib/utils/audiobook-matcher';
|
||||||
|
import { deduplicateAndCollectGroups } from '@/lib/utils/deduplicate-audiobooks';
|
||||||
|
import { persistDedupGroups } from '@/lib/services/works.service';
|
||||||
|
|
||||||
const logger = RMABLogger.create('API.Series.Detail');
|
const logger = RMABLogger.create('API.Series.Detail');
|
||||||
|
|
||||||
@@ -37,9 +39,11 @@ export async function GET(
|
|||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
logger.info(`Fetching series detail: ${asin}`);
|
const page = parseInt(request.nextUrl.searchParams.get('page') || '1', 10);
|
||||||
|
|
||||||
const detail = await scrapeSeriesPage(asin);
|
logger.info(`Fetching series detail: ${asin}, page ${page}`);
|
||||||
|
|
||||||
|
const detail = await scrapeSeriesPage(asin, page);
|
||||||
if (!detail) {
|
if (!detail) {
|
||||||
return NextResponse.json(
|
return NextResponse.json(
|
||||||
{ error: 'NotFound', message: 'Series not found' },
|
{ error: 'NotFound', message: 'Series not found' },
|
||||||
@@ -47,11 +51,19 @@ export async function GET(
|
|||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Deduplicate before enrichment to avoid wasted DB queries on duplicate entries
|
||||||
|
const { books: dedupedBooks, groups } = deduplicateAndCollectGroups(detail.books);
|
||||||
|
|
||||||
|
// Fire-and-forget: persist dedup groups to works table for cross-ASIN matching
|
||||||
|
if (groups.length > 0) {
|
||||||
|
persistDedupGroups(groups).catch(() => {});
|
||||||
|
}
|
||||||
|
|
||||||
// Enrich books with library availability and request status
|
// Enrich books with library availability and request status
|
||||||
const userId = currentUser.sub || undefined;
|
const userId = currentUser.sub || undefined;
|
||||||
const enrichedBooks = await enrichAudiobooksWithMatches(detail.books, userId);
|
const enrichedBooks = await enrichAudiobooksWithMatches(dedupedBooks, userId);
|
||||||
|
|
||||||
logger.info(`Series detail complete: "${detail.title}" (${enrichedBooks.length} books)`);
|
logger.info(`Series detail complete: "${detail.title}" (${enrichedBooks.length} books, page ${page})`);
|
||||||
|
|
||||||
return NextResponse.json({
|
return NextResponse.json({
|
||||||
success: true,
|
success: true,
|
||||||
@@ -59,6 +71,8 @@ export async function GET(
|
|||||||
...detail,
|
...detail,
|
||||||
books: enrichedBooks,
|
books: enrichedBooks,
|
||||||
},
|
},
|
||||||
|
hasMore: detail.hasMore,
|
||||||
|
page: detail.page,
|
||||||
});
|
});
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
logger.error('Failed to fetch series detail', {
|
logger.error('Failed to fetch series detail', {
|
||||||
|
|||||||
@@ -140,14 +140,15 @@ export async function POST(request: NextRequest) {
|
|||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
const normalizedAdminUsername = admin.username.trim().toLowerCase();
|
||||||
const hashedPassword = await bcrypt.hash(admin.password, 10);
|
const hashedPassword = await bcrypt.hash(admin.password, 10);
|
||||||
const encryptionService = getEncryptionService();
|
const encryptionService = getEncryptionService();
|
||||||
const encryptedPassword = encryptionService.encrypt(hashedPassword);
|
const encryptedPassword = encryptionService.encrypt(hashedPassword);
|
||||||
|
|
||||||
adminUser = await prisma.user.create({
|
adminUser = await prisma.user.create({
|
||||||
data: {
|
data: {
|
||||||
plexId: `local-${admin.username}`,
|
plexId: `local-${normalizedAdminUsername}`,
|
||||||
plexUsername: admin.username,
|
plexUsername: normalizedAdminUsername,
|
||||||
plexEmail: null,
|
plexEmail: null,
|
||||||
role: 'admin',
|
role: 'admin',
|
||||||
isSetupAdmin: true, // Mark as setup admin - role cannot be changed
|
isSetupAdmin: true, // Mark as setup admin - role cannot be changed
|
||||||
|
|||||||
@@ -0,0 +1,59 @@
|
|||||||
|
/**
|
||||||
|
* Component: User API Token Delete Route (self-service)
|
||||||
|
* Documentation: documentation/backend/services/api-tokens.md
|
||||||
|
*/
|
||||||
|
|
||||||
|
import { NextRequest, NextResponse } from 'next/server';
|
||||||
|
import { requireAuth, AuthenticatedRequest } from '@/lib/middleware/auth';
|
||||||
|
import { prisma } from '@/lib/db';
|
||||||
|
import { RMABLogger } from '@/lib/utils/logger';
|
||||||
|
import { checkApiTokenRevokeRateLimit } from '@/lib/utils/apiTokenRateLimit';
|
||||||
|
|
||||||
|
const logger = RMABLogger.create('API.User.ApiTokens');
|
||||||
|
|
||||||
|
/**
|
||||||
|
* DELETE /api/user/api-tokens/[id]
|
||||||
|
* Revoke (delete) one of the current user's own API tokens
|
||||||
|
*/
|
||||||
|
export async function DELETE(
|
||||||
|
request: NextRequest,
|
||||||
|
{ params }: { params: Promise<{ id: string }> }
|
||||||
|
) {
|
||||||
|
return requireAuth(request, async (req: AuthenticatedRequest) => {
|
||||||
|
try {
|
||||||
|
const rateLimit = checkApiTokenRevokeRateLimit(req.user!.id);
|
||||||
|
if (!rateLimit.allowed) {
|
||||||
|
return NextResponse.json(
|
||||||
|
{ error: 'Too many API token revoke attempts. Please try again later.' },
|
||||||
|
{
|
||||||
|
status: 429,
|
||||||
|
headers: {
|
||||||
|
'Retry-After': String(rateLimit.retryAfterSeconds),
|
||||||
|
},
|
||||||
|
}
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
const { id } = await params;
|
||||||
|
|
||||||
|
const token = await prisma.apiToken.findUnique({ where: { id } });
|
||||||
|
if (!token) {
|
||||||
|
return NextResponse.json({ error: 'Token not found' }, { status: 404 });
|
||||||
|
}
|
||||||
|
|
||||||
|
// Only allow deleting own tokens
|
||||||
|
if (token.userId !== req.user!.id) {
|
||||||
|
return NextResponse.json({ error: 'Token not found' }, { status: 404 });
|
||||||
|
}
|
||||||
|
|
||||||
|
await prisma.apiToken.delete({ where: { id } });
|
||||||
|
|
||||||
|
logger.info('User API token revoked', { tokenId: id, name: token.name, userId: req.user!.id });
|
||||||
|
|
||||||
|
return NextResponse.json({ success: true });
|
||||||
|
} catch (error) {
|
||||||
|
logger.error('Failed to revoke user API token', { error: error instanceof Error ? error.message : String(error) });
|
||||||
|
return NextResponse.json({ error: 'Failed to revoke API token' }, { status: 500 });
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
@@ -0,0 +1,141 @@
|
|||||||
|
/**
|
||||||
|
* Component: User API Token Routes (self-service)
|
||||||
|
* Documentation: documentation/backend/services/api-tokens.md
|
||||||
|
*/
|
||||||
|
|
||||||
|
import { NextRequest, NextResponse } from 'next/server';
|
||||||
|
import { requireAuth, AuthenticatedRequest } from '@/lib/middleware/auth';
|
||||||
|
import { prisma } from '@/lib/db';
|
||||||
|
import { RMABLogger } from '@/lib/utils/logger';
|
||||||
|
import { checkApiTokenCreateRateLimit } from '@/lib/utils/apiTokenRateLimit';
|
||||||
|
import { MAX_TOKENS_PER_USER } from '@/lib/constants/api-tokens';
|
||||||
|
import { generateApiToken } from '@/lib/utils/api-token';
|
||||||
|
import { z } from 'zod';
|
||||||
|
|
||||||
|
const logger = RMABLogger.create('API.User.ApiTokens');
|
||||||
|
|
||||||
|
const CreateTokenSchema = z.object({
|
||||||
|
name: z.string().min(1).max(100),
|
||||||
|
expiresAt: z.string().datetime().nullable().optional(),
|
||||||
|
});
|
||||||
|
|
||||||
|
/**
|
||||||
|
* GET /api/user/api-tokens
|
||||||
|
* List the current user's own API tokens
|
||||||
|
*/
|
||||||
|
export async function GET(request: NextRequest) {
|
||||||
|
return requireAuth(request, async (req: AuthenticatedRequest) => {
|
||||||
|
try {
|
||||||
|
const tokens = await prisma.apiToken.findMany({
|
||||||
|
where: { userId: req.user!.id },
|
||||||
|
orderBy: { createdAt: 'desc' },
|
||||||
|
});
|
||||||
|
|
||||||
|
const sanitized = tokens.map((t) => ({
|
||||||
|
id: t.id,
|
||||||
|
name: t.name,
|
||||||
|
tokenPrefix: t.tokenPrefix,
|
||||||
|
role: t.role,
|
||||||
|
lastUsedAt: t.lastUsedAt,
|
||||||
|
expiresAt: t.expiresAt,
|
||||||
|
createdAt: t.createdAt,
|
||||||
|
}));
|
||||||
|
|
||||||
|
return NextResponse.json({ tokens: sanitized });
|
||||||
|
} catch (error) {
|
||||||
|
logger.error('Failed to list user API tokens', { error: error instanceof Error ? error.message : String(error) });
|
||||||
|
return NextResponse.json({ error: 'Failed to list API tokens' }, { status: 500 });
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* POST /api/user/api-tokens
|
||||||
|
* Create a token for the current user with their own role. Returns full token ONCE.
|
||||||
|
*/
|
||||||
|
export async function POST(request: NextRequest) {
|
||||||
|
return requireAuth(request, async (req: AuthenticatedRequest) => {
|
||||||
|
try {
|
||||||
|
const rateLimit = checkApiTokenCreateRateLimit(req.user!.id);
|
||||||
|
if (!rateLimit.allowed) {
|
||||||
|
return NextResponse.json(
|
||||||
|
{ error: 'Too many API token create attempts. Please try again later.' },
|
||||||
|
{
|
||||||
|
status: 429,
|
||||||
|
headers: {
|
||||||
|
'Retry-After': String(rateLimit.retryAfterSeconds),
|
||||||
|
},
|
||||||
|
}
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
const body = await req.json();
|
||||||
|
const { name, expiresAt } = CreateTokenSchema.parse(body);
|
||||||
|
|
||||||
|
// Look up the user's actual role from the database
|
||||||
|
const user = await prisma.user.findUnique({
|
||||||
|
where: { id: req.user!.id },
|
||||||
|
select: { role: true },
|
||||||
|
});
|
||||||
|
|
||||||
|
if (!user) {
|
||||||
|
return NextResponse.json({ error: 'User not found' }, { status: 404 });
|
||||||
|
}
|
||||||
|
|
||||||
|
// Enforce per-user token cap (count only active, non-expired tokens)
|
||||||
|
const activeTokenCount = await prisma.apiToken.count({
|
||||||
|
where: {
|
||||||
|
userId: req.user!.id,
|
||||||
|
OR: [
|
||||||
|
{ expiresAt: null },
|
||||||
|
{ expiresAt: { gt: new Date() } },
|
||||||
|
],
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
if (activeTokenCount >= MAX_TOKENS_PER_USER) {
|
||||||
|
return NextResponse.json(
|
||||||
|
{ error: `Token limit reached. Users may have at most ${MAX_TOKENS_PER_USER} active API tokens.` },
|
||||||
|
{ status: 403 }
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Generate the token
|
||||||
|
const { fullToken, tokenHash, tokenPrefix } = generateApiToken();
|
||||||
|
|
||||||
|
const apiToken = await prisma.apiToken.create({
|
||||||
|
data: {
|
||||||
|
name,
|
||||||
|
tokenHash,
|
||||||
|
tokenPrefix,
|
||||||
|
role: user.role, // Always the user's own role
|
||||||
|
createdById: req.user!.id,
|
||||||
|
userId: req.user!.id, // Token acts as the current user
|
||||||
|
expiresAt: expiresAt ? new Date(expiresAt) : null,
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
logger.info('User API token created', { tokenId: apiToken.id, name, userId: req.user!.id });
|
||||||
|
|
||||||
|
return NextResponse.json({
|
||||||
|
token: {
|
||||||
|
id: apiToken.id,
|
||||||
|
name: apiToken.name,
|
||||||
|
tokenPrefix: apiToken.tokenPrefix,
|
||||||
|
role: apiToken.role,
|
||||||
|
expiresAt: apiToken.expiresAt,
|
||||||
|
createdAt: apiToken.createdAt,
|
||||||
|
},
|
||||||
|
fullToken,
|
||||||
|
}, { status: 201 });
|
||||||
|
} catch (error) {
|
||||||
|
logger.error('Failed to create user API token', { error: error instanceof Error ? error.message : String(error) });
|
||||||
|
|
||||||
|
if (error instanceof z.ZodError) {
|
||||||
|
return NextResponse.json({ error: 'Validation error', details: error.errors }, { status: 400 });
|
||||||
|
}
|
||||||
|
|
||||||
|
return NextResponse.json({ error: 'Failed to create API token' }, { status: 500 });
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
@@ -7,9 +7,15 @@ import { NextRequest, NextResponse } from 'next/server';
|
|||||||
import { requireAuth, AuthenticatedRequest } from '@/lib/middleware/auth';
|
import { requireAuth, AuthenticatedRequest } from '@/lib/middleware/auth';
|
||||||
import { prisma } from '@/lib/db';
|
import { prisma } from '@/lib/db';
|
||||||
import { RMABLogger } from '@/lib/utils/logger';
|
import { RMABLogger } from '@/lib/utils/logger';
|
||||||
|
import { getJobQueueService } from '@/lib/services/job-queue.service';
|
||||||
|
import { z } from 'zod';
|
||||||
|
|
||||||
const logger = RMABLogger.create('API.GoodreadsShelves');
|
const logger = RMABLogger.create('API.GoodreadsShelves');
|
||||||
|
|
||||||
|
const UpdateGoodreadsSchema = z.object({
|
||||||
|
rssUrl: z.string().url('Must be a valid URL'),
|
||||||
|
});
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* DELETE /api/user/goodreads-shelves/[id]
|
* DELETE /api/user/goodreads-shelves/[id]
|
||||||
* Remove a Goodreads shelf subscription (ownership check)
|
* Remove a Goodreads shelf subscription (ownership check)
|
||||||
@@ -48,3 +54,57 @@ export async function DELETE(
|
|||||||
}
|
}
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* PATCH /api/user/goodreads-shelves/[id]
|
||||||
|
* Update a Goodreads shelf subscription
|
||||||
|
*/
|
||||||
|
export async function PATCH(
|
||||||
|
request: NextRequest,
|
||||||
|
{ params }: { params: Promise<{ id: string }> }
|
||||||
|
) {
|
||||||
|
return requireAuth(request, async (req: AuthenticatedRequest) => {
|
||||||
|
try {
|
||||||
|
if (!req.user) {
|
||||||
|
return NextResponse.json({ error: 'Unauthorized' }, { status: 401 });
|
||||||
|
}
|
||||||
|
|
||||||
|
const { id } = await params;
|
||||||
|
const shelf = await prisma.goodreadsShelf.findUnique({ where: { id } });
|
||||||
|
|
||||||
|
if (!shelf) {
|
||||||
|
return NextResponse.json({ error: 'Shelf not found' }, { status: 404 });
|
||||||
|
}
|
||||||
|
|
||||||
|
if (shelf.userId !== req.user.id) {
|
||||||
|
return NextResponse.json({ error: 'Forbidden' }, { status: 403 });
|
||||||
|
}
|
||||||
|
|
||||||
|
const body = await request.json();
|
||||||
|
const { rssUrl } = UpdateGoodreadsSchema.parse(body);
|
||||||
|
|
||||||
|
// Force re-fetch by clearing metadata
|
||||||
|
const updated = await prisma.goodreadsShelf.update({
|
||||||
|
where: { id },
|
||||||
|
data: { rssUrl, lastSyncAt: null, bookCount: null, coverUrls: null },
|
||||||
|
});
|
||||||
|
|
||||||
|
try {
|
||||||
|
const jobQueue = getJobQueueService();
|
||||||
|
await jobQueue.addSyncShelvesJob(undefined, updated.id, 'goodreads', 0);
|
||||||
|
} catch (error) {
|
||||||
|
logger.error('Failed to trigger immediate list sync', {
|
||||||
|
error: error instanceof Error ? error.message : String(error),
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
return NextResponse.json({ success: true, shelf: updated });
|
||||||
|
} catch (error) {
|
||||||
|
if (error instanceof z.ZodError) {
|
||||||
|
return NextResponse.json({ error: 'ValidationError', details: error.errors }, { status: 400 });
|
||||||
|
}
|
||||||
|
logger.error('Failed to update shelf', { error: error instanceof Error ? error.message : String(error) });
|
||||||
|
return NextResponse.json({ error: 'Failed to update shelf' }, { status: 500 });
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|||||||
@@ -139,8 +139,8 @@ export async function POST(request: NextRequest) {
|
|||||||
// Trigger immediate sync for this shelf (unlimited lookups, process all books)
|
// Trigger immediate sync for this shelf (unlimited lookups, process all books)
|
||||||
try {
|
try {
|
||||||
const jobQueue = getJobQueueService();
|
const jobQueue = getJobQueueService();
|
||||||
await jobQueue.addSyncGoodreadsShelvesJob(undefined, shelf.id, 0);
|
await jobQueue.addSyncShelvesJob(undefined, shelf.id, 'goodreads', 0);
|
||||||
logger.info(`Triggered immediate sync for shelf "${shelfName}" (${shelf.id})`);
|
logger.info(`Triggered immediate sync for Goodreads shelf "${shelfName}" (${shelf.id})`);
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
logger.error('Failed to trigger immediate shelf sync', { error: error instanceof Error ? error.message : String(error) });
|
logger.error('Failed to trigger immediate shelf sync', { error: error instanceof Error ? error.message : String(error) });
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -0,0 +1,177 @@
|
|||||||
|
/**
|
||||||
|
* Component: Hardcover Shelf Delete Route
|
||||||
|
* Documentation: documentation/backend/services/hardcover-sync.md
|
||||||
|
*/
|
||||||
|
|
||||||
|
import { NextRequest, NextResponse } from 'next/server';
|
||||||
|
import { requireAuth, AuthenticatedRequest } from '@/lib/middleware/auth';
|
||||||
|
import { prisma } from '@/lib/db';
|
||||||
|
import { RMABLogger } from '@/lib/utils/logger';
|
||||||
|
import { getJobQueueService } from '@/lib/services/job-queue.service';
|
||||||
|
import { getEncryptionService } from '@/lib/services/encryption.service';
|
||||||
|
import { fetchHardcoverList } from '@/lib/services/hardcover-api.service';
|
||||||
|
import { z } from 'zod';
|
||||||
|
|
||||||
|
const logger = RMABLogger.create('API.HardcoverShelves');
|
||||||
|
|
||||||
|
const UpdateHardcoverSchema = z.object({
|
||||||
|
listId: z.string().min(1, 'List ID is required').optional(),
|
||||||
|
apiToken: z.string().optional(),
|
||||||
|
});
|
||||||
|
|
||||||
|
/**
|
||||||
|
* DELETE /api/user/hardcover-shelves/[id]
|
||||||
|
* Remove a Hardcover shelf subscription (ownership check)
|
||||||
|
*/
|
||||||
|
export async function DELETE(
|
||||||
|
request: NextRequest,
|
||||||
|
{ params }: { params: Promise<{ id: string }> },
|
||||||
|
) {
|
||||||
|
return requireAuth(request, async (req: AuthenticatedRequest) => {
|
||||||
|
try {
|
||||||
|
if (!req.user) {
|
||||||
|
return NextResponse.json({ error: 'Unauthorized' }, { status: 401 });
|
||||||
|
}
|
||||||
|
|
||||||
|
const { id } = await params;
|
||||||
|
|
||||||
|
const shelf = await prisma.hardcoverShelf.findUnique({
|
||||||
|
where: { id },
|
||||||
|
});
|
||||||
|
|
||||||
|
if (!shelf) {
|
||||||
|
return NextResponse.json({ error: 'List not found' }, { status: 404 });
|
||||||
|
}
|
||||||
|
|
||||||
|
// Ownership check
|
||||||
|
if (shelf.userId !== req.user.id) {
|
||||||
|
return NextResponse.json({ error: 'Forbidden' }, { status: 403 });
|
||||||
|
}
|
||||||
|
|
||||||
|
await prisma.hardcoverShelf.delete({ where: { id } });
|
||||||
|
|
||||||
|
return NextResponse.json({ success: true });
|
||||||
|
} catch (error) {
|
||||||
|
logger.error('Failed to delete list', {
|
||||||
|
error: error instanceof Error ? error.message : String(error),
|
||||||
|
});
|
||||||
|
return NextResponse.json(
|
||||||
|
{ error: 'Failed to delete list' },
|
||||||
|
{ status: 500 },
|
||||||
|
);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* PATCH /api/user/hardcover-shelves/[id]
|
||||||
|
* Update a Hardcover shelf subscription
|
||||||
|
*/
|
||||||
|
export async function PATCH(
|
||||||
|
request: NextRequest,
|
||||||
|
{ params }: { params: Promise<{ id: string }> },
|
||||||
|
) {
|
||||||
|
return requireAuth(request, async (req: AuthenticatedRequest) => {
|
||||||
|
try {
|
||||||
|
if (!req.user) {
|
||||||
|
return NextResponse.json({ error: 'Unauthorized' }, { status: 401 });
|
||||||
|
}
|
||||||
|
|
||||||
|
const { id } = await params;
|
||||||
|
const shelf = await prisma.hardcoverShelf.findUnique({ where: { id } });
|
||||||
|
|
||||||
|
if (!shelf) {
|
||||||
|
return NextResponse.json({ error: 'List not found' }, { status: 404 });
|
||||||
|
}
|
||||||
|
|
||||||
|
if (shelf.userId !== req.user.id) {
|
||||||
|
return NextResponse.json({ error: 'Forbidden' }, { status: 403 });
|
||||||
|
}
|
||||||
|
|
||||||
|
const body = await request.json();
|
||||||
|
const { listId, apiToken } = UpdateHardcoverSchema.parse(body);
|
||||||
|
|
||||||
|
const updateData: { listId?: string; apiToken?: string; lastSyncAt?: null; bookCount?: null; coverUrls?: null } = {};
|
||||||
|
let needsResync = false;
|
||||||
|
|
||||||
|
let cleanedToken: string | undefined;
|
||||||
|
if (apiToken && apiToken.trim() !== '') {
|
||||||
|
cleanedToken = apiToken.trim().toLowerCase().startsWith('bearer ')
|
||||||
|
? apiToken.trim().slice(7).trim()
|
||||||
|
: apiToken.trim();
|
||||||
|
}
|
||||||
|
|
||||||
|
const newListId = (listId && listId !== shelf.listId) ? listId : undefined;
|
||||||
|
|
||||||
|
// Validate token/listId by fetching the list before saving
|
||||||
|
if (cleanedToken || newListId) {
|
||||||
|
const encryptionService = getEncryptionService();
|
||||||
|
let tokenToTest = cleanedToken || shelf.apiToken;
|
||||||
|
if (!cleanedToken) {
|
||||||
|
try {
|
||||||
|
if (encryptionService.isEncryptedFormat(shelf.apiToken)) {
|
||||||
|
tokenToTest = encryptionService.decrypt(shelf.apiToken);
|
||||||
|
}
|
||||||
|
} catch {
|
||||||
|
// Decryption failed, fall back to raw token
|
||||||
|
}
|
||||||
|
}
|
||||||
|
const listIdToTest = newListId || shelf.listId;
|
||||||
|
|
||||||
|
try {
|
||||||
|
await fetchHardcoverList(tokenToTest, listIdToTest);
|
||||||
|
} catch (error) {
|
||||||
|
return NextResponse.json(
|
||||||
|
{
|
||||||
|
error: 'InvalidHardcoverList',
|
||||||
|
message: `Could not fetch the Hardcover list. Check your Token and List ID: ${error instanceof Error ? error.message : 'Unknown error'}`,
|
||||||
|
},
|
||||||
|
{ status: 400 },
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (newListId) {
|
||||||
|
updateData.listId = newListId;
|
||||||
|
needsResync = true;
|
||||||
|
}
|
||||||
|
if (cleanedToken) {
|
||||||
|
updateData.apiToken = encryptionService.encrypt(cleanedToken);
|
||||||
|
needsResync = true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// If we are forcing a resync due to a change, clear metadata
|
||||||
|
if (needsResync) {
|
||||||
|
updateData.lastSyncAt = null;
|
||||||
|
updateData.bookCount = null;
|
||||||
|
updateData.coverUrls = null;
|
||||||
|
}
|
||||||
|
|
||||||
|
const updated = await prisma.hardcoverShelf.update({
|
||||||
|
where: { id },
|
||||||
|
data: updateData,
|
||||||
|
});
|
||||||
|
|
||||||
|
if (needsResync) {
|
||||||
|
try {
|
||||||
|
const jobQueue = getJobQueueService();
|
||||||
|
await jobQueue.addSyncShelvesJob(undefined, updated.id, 'hardcover', 0);
|
||||||
|
} catch (error) {
|
||||||
|
logger.error('Failed to trigger immediate list sync', {
|
||||||
|
error: error instanceof Error ? error.message : String(error),
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return NextResponse.json({ success: true, shelf: updated });
|
||||||
|
} catch (error) {
|
||||||
|
if (error instanceof z.ZodError) {
|
||||||
|
return NextResponse.json({ error: 'ValidationError', details: error.errors }, { status: 400 });
|
||||||
|
}
|
||||||
|
logger.error('Failed to update list', {
|
||||||
|
error: error instanceof Error ? error.message : String(error),
|
||||||
|
});
|
||||||
|
return NextResponse.json({ error: 'Failed to update list' }, { status: 500 });
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
@@ -0,0 +1,195 @@
|
|||||||
|
/**
|
||||||
|
* Component: Hardcover Shelves API Routes
|
||||||
|
* Documentation: documentation/backend/services/hardcover-sync.md
|
||||||
|
*/
|
||||||
|
|
||||||
|
import { NextRequest, NextResponse } from 'next/server';
|
||||||
|
import { requireAuth, AuthenticatedRequest } from '@/lib/middleware/auth';
|
||||||
|
import { prisma } from '@/lib/db';
|
||||||
|
import { fetchHardcoverList } from '@/lib/services/hardcover-api.service';
|
||||||
|
import { getJobQueueService } from '@/lib/services/job-queue.service';
|
||||||
|
import { getEncryptionService } from '@/lib/services/encryption.service';
|
||||||
|
import { z } from 'zod';
|
||||||
|
import { RMABLogger } from '@/lib/utils/logger';
|
||||||
|
import { processBooks } from '@/lib/utils/shelf-helpers';
|
||||||
|
|
||||||
|
const logger = RMABLogger.create('API.HardcoverShelves');
|
||||||
|
|
||||||
|
const AddShelfSchema = z.object({
|
||||||
|
listId: z.string().min(1, { message: 'List ID is required' }),
|
||||||
|
apiToken: z.string().min(1, { message: 'API Token is required' }),
|
||||||
|
});
|
||||||
|
|
||||||
|
/**
|
||||||
|
* GET /api/user/hardcover-shelves
|
||||||
|
* List the current user's Hardcover lists with book counts and covers
|
||||||
|
*/
|
||||||
|
export async function GET(request: NextRequest) {
|
||||||
|
return requireAuth(request, async (req: AuthenticatedRequest) => {
|
||||||
|
try {
|
||||||
|
if (!req.user) {
|
||||||
|
return NextResponse.json({ error: 'Unauthorized' }, { status: 401 });
|
||||||
|
}
|
||||||
|
|
||||||
|
const shelves = await prisma.hardcoverShelf.findMany({
|
||||||
|
where: { userId: req.user.id },
|
||||||
|
orderBy: { createdAt: 'desc' },
|
||||||
|
});
|
||||||
|
|
||||||
|
const shelvesWithMeta = shelves.map((shelf) => {
|
||||||
|
const books = processBooks(shelf.coverUrls);
|
||||||
|
|
||||||
|
return {
|
||||||
|
id: shelf.id,
|
||||||
|
name: shelf.name,
|
||||||
|
listId: shelf.listId,
|
||||||
|
lastSyncAt: shelf.lastSyncAt,
|
||||||
|
createdAt: shelf.createdAt,
|
||||||
|
bookCount: shelf.bookCount ?? null,
|
||||||
|
books,
|
||||||
|
};
|
||||||
|
});
|
||||||
|
|
||||||
|
return NextResponse.json({ success: true, shelves: shelvesWithMeta });
|
||||||
|
} catch (error) {
|
||||||
|
logger.error('Failed to list Hardcover lists', {
|
||||||
|
error: error instanceof Error ? error.message : String(error),
|
||||||
|
});
|
||||||
|
return NextResponse.json(
|
||||||
|
{ error: 'Failed to list Hardcover lists' },
|
||||||
|
{ status: 500 },
|
||||||
|
);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* POST /api/user/hardcover-shelves
|
||||||
|
* Add a new Hardcover list subscription
|
||||||
|
*/
|
||||||
|
export async function POST(request: NextRequest) {
|
||||||
|
return requireAuth(request, async (req: AuthenticatedRequest) => {
|
||||||
|
try {
|
||||||
|
if (!req.user) {
|
||||||
|
return NextResponse.json({ error: 'Unauthorized' }, { status: 401 });
|
||||||
|
}
|
||||||
|
|
||||||
|
const body = await req.json();
|
||||||
|
let { listId, apiToken } = AddShelfSchema.parse(body);
|
||||||
|
|
||||||
|
// Clean up token in case user pasted "Bearer " prefix
|
||||||
|
apiToken = apiToken.trim();
|
||||||
|
if (apiToken.toLowerCase().startsWith('bearer ')) {
|
||||||
|
apiToken = apiToken.slice(7).trim();
|
||||||
|
}
|
||||||
|
|
||||||
|
// Check for duplicate
|
||||||
|
const existing = await prisma.hardcoverShelf.findUnique({
|
||||||
|
where: { userId_listId: { userId: req.user.id, listId } },
|
||||||
|
});
|
||||||
|
|
||||||
|
if (existing) {
|
||||||
|
return NextResponse.json(
|
||||||
|
{
|
||||||
|
error: 'DuplicateShelf',
|
||||||
|
message: 'You have already added this list',
|
||||||
|
},
|
||||||
|
{ status: 409 },
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Validate by fetching the Hardcover GraphQL feed
|
||||||
|
let listName: string;
|
||||||
|
let bookCount: number;
|
||||||
|
let initialBooks: {
|
||||||
|
coverUrl: string;
|
||||||
|
asin: null;
|
||||||
|
title: string;
|
||||||
|
author: string;
|
||||||
|
}[] = [];
|
||||||
|
try {
|
||||||
|
const fetchedData = await fetchHardcoverList(apiToken, listId);
|
||||||
|
listName = fetchedData.listName;
|
||||||
|
bookCount = fetchedData.books.length;
|
||||||
|
initialBooks = fetchedData.books
|
||||||
|
.filter((b) => b.coverUrl)
|
||||||
|
.slice(0, 8)
|
||||||
|
.map((b) => ({
|
||||||
|
coverUrl: b.coverUrl!,
|
||||||
|
asin: null,
|
||||||
|
title: b.title,
|
||||||
|
author: b.author,
|
||||||
|
}));
|
||||||
|
} catch (error) {
|
||||||
|
return NextResponse.json(
|
||||||
|
{
|
||||||
|
error: 'InvalidHardcoverList',
|
||||||
|
message: `Could not fetch the Hardcover list. Check your Token and List ID: ${error instanceof Error ? error.message : 'Unknown error'}`,
|
||||||
|
},
|
||||||
|
{ status: 400 },
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
const encryptionService = getEncryptionService();
|
||||||
|
const encryptedToken = encryptionService.encrypt(apiToken);
|
||||||
|
|
||||||
|
const shelf = await prisma.hardcoverShelf.create({
|
||||||
|
data: {
|
||||||
|
userId: req.user.id,
|
||||||
|
name: listName,
|
||||||
|
listId,
|
||||||
|
apiToken: encryptedToken,
|
||||||
|
bookCount,
|
||||||
|
coverUrls:
|
||||||
|
initialBooks.length > 0 ? JSON.stringify(initialBooks) : null,
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
// Trigger immediate sync for this shelf (unlimited lookups, process all books)
|
||||||
|
try {
|
||||||
|
const jobQueue = getJobQueueService();
|
||||||
|
await jobQueue.addSyncShelvesJob(undefined, shelf.id, 'hardcover', 0);
|
||||||
|
logger.info(
|
||||||
|
`Triggered immediate sync for Hardcover list "${listName}" (${shelf.id})`,
|
||||||
|
);
|
||||||
|
} catch (error) {
|
||||||
|
logger.error('Failed to trigger immediate list sync', {
|
||||||
|
error: error instanceof Error ? error.message : String(error),
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
return NextResponse.json(
|
||||||
|
{
|
||||||
|
success: true,
|
||||||
|
shelf: {
|
||||||
|
id: shelf.id,
|
||||||
|
name: shelf.name,
|
||||||
|
listId: shelf.listId,
|
||||||
|
lastSyncAt: shelf.lastSyncAt,
|
||||||
|
createdAt: shelf.createdAt,
|
||||||
|
bookCount: shelf.bookCount,
|
||||||
|
books: initialBooks,
|
||||||
|
},
|
||||||
|
bookCount,
|
||||||
|
},
|
||||||
|
{ status: 201 },
|
||||||
|
);
|
||||||
|
} catch (error) {
|
||||||
|
logger.error('Failed to add Hardcover list', {
|
||||||
|
error: error instanceof Error ? error.message : String(error),
|
||||||
|
});
|
||||||
|
|
||||||
|
if (error instanceof z.ZodError) {
|
||||||
|
return NextResponse.json(
|
||||||
|
{ error: 'ValidationError', details: error.errors },
|
||||||
|
{ status: 400 },
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
return NextResponse.json(
|
||||||
|
{ error: 'Failed to add Hardcover list' },
|
||||||
|
{ status: 500 },
|
||||||
|
);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
@@ -0,0 +1,202 @@
|
|||||||
|
/**
|
||||||
|
* Component: User Home Sections API Route
|
||||||
|
* Documentation: documentation/features/home-sections.md
|
||||||
|
*
|
||||||
|
* Per-user configurable home page sections.
|
||||||
|
* GET returns sections + next refresh time.
|
||||||
|
* PUT saves full section config (delete-and-recreate in transaction).
|
||||||
|
*/
|
||||||
|
|
||||||
|
import { NextRequest, NextResponse } from 'next/server';
|
||||||
|
import { requireAuth, AuthenticatedRequest } from '@/lib/middleware/auth';
|
||||||
|
import { prisma } from '@/lib/db';
|
||||||
|
import { z } from 'zod';
|
||||||
|
import { RMABLogger } from '@/lib/utils/logger';
|
||||||
|
|
||||||
|
const logger = RMABLogger.create('API.User.HomeSections');
|
||||||
|
|
||||||
|
const MAX_SECTIONS = 10;
|
||||||
|
|
||||||
|
const VALID_SECTION_TYPES = ['popular', 'new_releases', 'category'] as const;
|
||||||
|
|
||||||
|
const SectionSchema = z.object({
|
||||||
|
sectionType: z.enum(VALID_SECTION_TYPES),
|
||||||
|
categoryId: z.string().optional().nullable(),
|
||||||
|
categoryName: z.string().optional().nullable(),
|
||||||
|
sortOrder: z.number().int().min(0),
|
||||||
|
});
|
||||||
|
|
||||||
|
const PutBodySchema = z.object({
|
||||||
|
sections: z.array(SectionSchema).max(MAX_SECTIONS),
|
||||||
|
});
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Create default home sections for a new user (Popular + New Releases).
|
||||||
|
*/
|
||||||
|
async function ensureDefaultSections(userId: string) {
|
||||||
|
const existing = await prisma.userHomeSection.findMany({
|
||||||
|
where: { userId },
|
||||||
|
select: { id: true },
|
||||||
|
take: 1,
|
||||||
|
});
|
||||||
|
|
||||||
|
if (existing.length > 0) return;
|
||||||
|
|
||||||
|
await prisma.userHomeSection.createMany({
|
||||||
|
data: [
|
||||||
|
{ userId, sectionType: 'popular', sortOrder: 0 },
|
||||||
|
{ userId, sectionType: 'new_releases', sortOrder: 1 },
|
||||||
|
],
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* GET /api/user/home-sections
|
||||||
|
* Returns the user's configured home sections + next scheduled refresh time.
|
||||||
|
*/
|
||||||
|
export async function GET(request: NextRequest) {
|
||||||
|
return requireAuth(request, async (req: AuthenticatedRequest) => {
|
||||||
|
try {
|
||||||
|
if (!req.user) {
|
||||||
|
return NextResponse.json({ error: 'Unauthorized' }, { status: 401 });
|
||||||
|
}
|
||||||
|
|
||||||
|
await ensureDefaultSections(req.user.id);
|
||||||
|
|
||||||
|
const sections = await prisma.userHomeSection.findMany({
|
||||||
|
where: { userId: req.user.id },
|
||||||
|
orderBy: { sortOrder: 'asc' },
|
||||||
|
});
|
||||||
|
|
||||||
|
// Get next refresh time from scheduled jobs
|
||||||
|
let nextRefresh: string | null = null;
|
||||||
|
try {
|
||||||
|
const scheduledJob = await prisma.scheduledJob.findFirst({
|
||||||
|
where: { type: 'audible_refresh', enabled: true },
|
||||||
|
select: { nextRun: true },
|
||||||
|
});
|
||||||
|
nextRefresh = scheduledJob?.nextRun?.toISOString() || null;
|
||||||
|
} catch {
|
||||||
|
// Non-critical — just omit nextRefresh
|
||||||
|
}
|
||||||
|
|
||||||
|
return NextResponse.json({
|
||||||
|
success: true,
|
||||||
|
sections: sections.map((s) => ({
|
||||||
|
id: s.id,
|
||||||
|
sectionType: s.sectionType,
|
||||||
|
categoryId: s.categoryId,
|
||||||
|
categoryName: s.categoryName,
|
||||||
|
sortOrder: s.sortOrder,
|
||||||
|
})),
|
||||||
|
nextRefresh,
|
||||||
|
});
|
||||||
|
} catch (error) {
|
||||||
|
logger.error('Failed to get home sections', {
|
||||||
|
error: error instanceof Error ? error.message : String(error),
|
||||||
|
});
|
||||||
|
return NextResponse.json(
|
||||||
|
{ error: 'FetchError', message: 'Failed to fetch home sections' },
|
||||||
|
{ status: 500 }
|
||||||
|
);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* PUT /api/user/home-sections
|
||||||
|
* Replaces all home sections for the user (delete-and-recreate in transaction).
|
||||||
|
* Validates: max 10 sections, no duplicate sections, category sections need categoryId.
|
||||||
|
*/
|
||||||
|
export async function PUT(request: NextRequest) {
|
||||||
|
return requireAuth(request, async (req: AuthenticatedRequest) => {
|
||||||
|
try {
|
||||||
|
if (!req.user) {
|
||||||
|
return NextResponse.json({ error: 'Unauthorized' }, { status: 401 });
|
||||||
|
}
|
||||||
|
|
||||||
|
const body = await req.json();
|
||||||
|
const { sections } = PutBodySchema.parse(body);
|
||||||
|
|
||||||
|
// Validate category sections have categoryId
|
||||||
|
for (const section of sections) {
|
||||||
|
if (section.sectionType === 'category' && !section.categoryId) {
|
||||||
|
return NextResponse.json(
|
||||||
|
{ error: 'ValidationError', message: 'Category sections require a categoryId' },
|
||||||
|
{ status: 400 }
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Check for duplicate section types (only one popular, one new_releases, unique categories)
|
||||||
|
const seen = new Set<string>();
|
||||||
|
for (const section of sections) {
|
||||||
|
const key =
|
||||||
|
section.sectionType === 'category'
|
||||||
|
? `category:${section.categoryId}`
|
||||||
|
: section.sectionType;
|
||||||
|
if (seen.has(key)) {
|
||||||
|
return NextResponse.json(
|
||||||
|
{ error: 'ValidationError', message: `Duplicate section: ${key}` },
|
||||||
|
{ status: 400 }
|
||||||
|
);
|
||||||
|
}
|
||||||
|
seen.add(key);
|
||||||
|
}
|
||||||
|
|
||||||
|
const userId = req.user.id;
|
||||||
|
|
||||||
|
// Delete-and-recreate in a transaction
|
||||||
|
await prisma.$transaction(async (tx) => {
|
||||||
|
await tx.userHomeSection.deleteMany({ where: { userId } });
|
||||||
|
|
||||||
|
if (sections.length > 0) {
|
||||||
|
await tx.userHomeSection.createMany({
|
||||||
|
data: sections.map((s, idx) => ({
|
||||||
|
userId,
|
||||||
|
sectionType: s.sectionType,
|
||||||
|
categoryId: s.sectionType === 'category' ? s.categoryId : null,
|
||||||
|
categoryName: s.sectionType === 'category' ? s.categoryName : null,
|
||||||
|
sortOrder: idx,
|
||||||
|
})),
|
||||||
|
});
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
// Return the saved sections
|
||||||
|
const saved = await prisma.userHomeSection.findMany({
|
||||||
|
where: { userId },
|
||||||
|
orderBy: { sortOrder: 'asc' },
|
||||||
|
});
|
||||||
|
|
||||||
|
logger.info(`User ${userId} updated home sections (${saved.length} sections)`);
|
||||||
|
|
||||||
|
return NextResponse.json({
|
||||||
|
success: true,
|
||||||
|
sections: saved.map((s) => ({
|
||||||
|
id: s.id,
|
||||||
|
sectionType: s.sectionType,
|
||||||
|
categoryId: s.categoryId,
|
||||||
|
categoryName: s.categoryName,
|
||||||
|
sortOrder: s.sortOrder,
|
||||||
|
})),
|
||||||
|
});
|
||||||
|
} catch (error) {
|
||||||
|
logger.error('Failed to save home sections', {
|
||||||
|
error: error instanceof Error ? error.message : String(error),
|
||||||
|
});
|
||||||
|
|
||||||
|
if (error instanceof z.ZodError) {
|
||||||
|
return NextResponse.json(
|
||||||
|
{ error: 'ValidationError', details: error.errors },
|
||||||
|
{ status: 400 }
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
return NextResponse.json(
|
||||||
|
{ error: 'SaveError', message: 'Failed to save home sections' },
|
||||||
|
{ status: 500 }
|
||||||
|
);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
@@ -0,0 +1,73 @@
|
|||||||
|
/**
|
||||||
|
* Component: Combined Shelves API Routes
|
||||||
|
* Documentation: documentation/backend/services/goodreads-sync.md
|
||||||
|
*/
|
||||||
|
|
||||||
|
import { NextRequest, NextResponse } from 'next/server';
|
||||||
|
import { requireAuth, AuthenticatedRequest } from '@/lib/middleware/auth';
|
||||||
|
import { prisma } from '@/lib/db';
|
||||||
|
import { RMABLogger } from '@/lib/utils/logger';
|
||||||
|
import { processBooks } from '@/lib/utils/shelf-helpers';
|
||||||
|
|
||||||
|
const logger = RMABLogger.create('API.Shelves');
|
||||||
|
|
||||||
|
/**
|
||||||
|
* GET /api/user/shelves
|
||||||
|
* List the current user's shelves (Goodreads, Hardcover) with book counts and covers
|
||||||
|
*/
|
||||||
|
export async function GET(request: NextRequest) {
|
||||||
|
return requireAuth(request, async (req: AuthenticatedRequest) => {
|
||||||
|
try {
|
||||||
|
if (!req.user) {
|
||||||
|
return NextResponse.json({ error: 'Unauthorized' }, { status: 401 });
|
||||||
|
}
|
||||||
|
|
||||||
|
const [goodreads, hardcover] = await Promise.all([
|
||||||
|
prisma.goodreadsShelf.findMany({
|
||||||
|
where: { userId: req.user.id },
|
||||||
|
orderBy: { createdAt: 'desc' },
|
||||||
|
}),
|
||||||
|
prisma.hardcoverShelf.findMany({
|
||||||
|
where: { userId: req.user.id },
|
||||||
|
orderBy: { createdAt: 'desc' },
|
||||||
|
}),
|
||||||
|
]);
|
||||||
|
|
||||||
|
const combined = [
|
||||||
|
...goodreads.map((s) => ({
|
||||||
|
id: s.id,
|
||||||
|
type: 'goodreads',
|
||||||
|
name: s.name,
|
||||||
|
sourceId: s.rssUrl,
|
||||||
|
lastSyncAt: s.lastSyncAt,
|
||||||
|
createdAt: s.createdAt,
|
||||||
|
bookCount: s.bookCount ?? null,
|
||||||
|
books: processBooks(s.coverUrls),
|
||||||
|
})),
|
||||||
|
...hardcover.map((s) => ({
|
||||||
|
id: s.id,
|
||||||
|
type: 'hardcover',
|
||||||
|
name: s.name,
|
||||||
|
sourceId: s.listId,
|
||||||
|
lastSyncAt: s.lastSyncAt,
|
||||||
|
createdAt: s.createdAt,
|
||||||
|
bookCount: s.bookCount ?? null,
|
||||||
|
books: processBooks(s.coverUrls),
|
||||||
|
})),
|
||||||
|
].sort(
|
||||||
|
(a, b) =>
|
||||||
|
new Date(b.createdAt).getTime() - new Date(a.createdAt).getTime(),
|
||||||
|
);
|
||||||
|
|
||||||
|
return NextResponse.json({ success: true, shelves: combined });
|
||||||
|
} catch (error) {
|
||||||
|
logger.error('Failed to list shelves', {
|
||||||
|
error: error instanceof Error ? error.message : String(error),
|
||||||
|
});
|
||||||
|
return NextResponse.json(
|
||||||
|
{ error: 'Failed to list shelves' },
|
||||||
|
{ status: 500 },
|
||||||
|
);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
@@ -0,0 +1,52 @@
|
|||||||
|
/**
|
||||||
|
* Component: Watched Author Delete Route
|
||||||
|
* Documentation: documentation/features/watched-lists.md
|
||||||
|
*/
|
||||||
|
|
||||||
|
import { NextRequest, NextResponse } from 'next/server';
|
||||||
|
import { requireAuth, AuthenticatedRequest } from '@/lib/middleware/auth';
|
||||||
|
import { prisma } from '@/lib/db';
|
||||||
|
import { RMABLogger } from '@/lib/utils/logger';
|
||||||
|
|
||||||
|
const logger = RMABLogger.create('API.WatchedAuthors');
|
||||||
|
|
||||||
|
/**
|
||||||
|
* DELETE /api/user/watched-authors/[id]
|
||||||
|
* Remove an author from the user's watch list (ownership check)
|
||||||
|
*/
|
||||||
|
export async function DELETE(
|
||||||
|
request: NextRequest,
|
||||||
|
{ params }: { params: Promise<{ id: string }> }
|
||||||
|
) {
|
||||||
|
return requireAuth(request, async (req: AuthenticatedRequest) => {
|
||||||
|
try {
|
||||||
|
if (!req.user) {
|
||||||
|
return NextResponse.json({ error: 'Unauthorized' }, { status: 401 });
|
||||||
|
}
|
||||||
|
|
||||||
|
const { id } = await params;
|
||||||
|
|
||||||
|
const watched = await prisma.watchedAuthor.findUnique({
|
||||||
|
where: { id },
|
||||||
|
});
|
||||||
|
|
||||||
|
if (!watched) {
|
||||||
|
return NextResponse.json({ error: 'Watched author not found' }, { status: 404 });
|
||||||
|
}
|
||||||
|
|
||||||
|
// Ownership check
|
||||||
|
if (watched.userId !== req.user.id) {
|
||||||
|
return NextResponse.json({ error: 'Forbidden' }, { status: 403 });
|
||||||
|
}
|
||||||
|
|
||||||
|
await prisma.watchedAuthor.delete({ where: { id } });
|
||||||
|
|
||||||
|
logger.info(`User ${req.user.id} stopped watching author "${watched.authorName}" (${watched.authorAsin})`);
|
||||||
|
|
||||||
|
return NextResponse.json({ success: true });
|
||||||
|
} catch (error) {
|
||||||
|
logger.error('Failed to delete watched author', { error: error instanceof Error ? error.message : String(error) });
|
||||||
|
return NextResponse.json({ error: 'Failed to delete watched author' }, { status: 500 });
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
@@ -0,0 +1,125 @@
|
|||||||
|
/**
|
||||||
|
* Component: Watched Authors API Routes
|
||||||
|
* Documentation: documentation/features/watched-lists.md
|
||||||
|
*/
|
||||||
|
|
||||||
|
import { NextRequest, NextResponse } from 'next/server';
|
||||||
|
import { requireAuth, AuthenticatedRequest } from '@/lib/middleware/auth';
|
||||||
|
import { prisma } from '@/lib/db';
|
||||||
|
import { getJobQueueService } from '@/lib/services/job-queue.service';
|
||||||
|
import { z } from 'zod';
|
||||||
|
import { RMABLogger } from '@/lib/utils/logger';
|
||||||
|
|
||||||
|
const logger = RMABLogger.create('API.WatchedAuthors');
|
||||||
|
|
||||||
|
const AddWatchedAuthorSchema = z.object({
|
||||||
|
authorAsin: z.string().regex(/^[A-Z0-9]{10}$/, 'Invalid author ASIN'),
|
||||||
|
authorName: z.string().min(1).max(500),
|
||||||
|
coverArtUrl: z.string().url().optional(),
|
||||||
|
});
|
||||||
|
|
||||||
|
/**
|
||||||
|
* GET /api/user/watched-authors
|
||||||
|
* List the current user's watched authors
|
||||||
|
*/
|
||||||
|
export async function GET(request: NextRequest) {
|
||||||
|
return requireAuth(request, async (req: AuthenticatedRequest) => {
|
||||||
|
try {
|
||||||
|
if (!req.user) {
|
||||||
|
return NextResponse.json({ error: 'Unauthorized' }, { status: 401 });
|
||||||
|
}
|
||||||
|
|
||||||
|
const authors = await prisma.watchedAuthor.findMany({
|
||||||
|
where: { userId: req.user.id },
|
||||||
|
orderBy: { createdAt: 'desc' },
|
||||||
|
});
|
||||||
|
|
||||||
|
return NextResponse.json({
|
||||||
|
success: true,
|
||||||
|
authors: authors.map((a) => ({
|
||||||
|
id: a.id,
|
||||||
|
authorAsin: a.authorAsin,
|
||||||
|
authorName: a.authorName,
|
||||||
|
coverArtUrl: a.coverArtUrl,
|
||||||
|
lastCheckedAt: a.lastCheckedAt,
|
||||||
|
createdAt: a.createdAt,
|
||||||
|
})),
|
||||||
|
});
|
||||||
|
} catch (error) {
|
||||||
|
logger.error('Failed to list watched authors', { error: error instanceof Error ? error.message : String(error) });
|
||||||
|
return NextResponse.json({ error: 'Failed to list watched authors' }, { status: 500 });
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* POST /api/user/watched-authors
|
||||||
|
* Add an author to the user's watch list
|
||||||
|
*/
|
||||||
|
export async function POST(request: NextRequest) {
|
||||||
|
return requireAuth(request, async (req: AuthenticatedRequest) => {
|
||||||
|
try {
|
||||||
|
if (!req.user) {
|
||||||
|
return NextResponse.json({ error: 'Unauthorized' }, { status: 401 });
|
||||||
|
}
|
||||||
|
|
||||||
|
const body = await req.json();
|
||||||
|
const { authorAsin, authorName, coverArtUrl } = AddWatchedAuthorSchema.parse(body);
|
||||||
|
|
||||||
|
// Check for duplicate
|
||||||
|
const existing = await prisma.watchedAuthor.findUnique({
|
||||||
|
where: { userId_authorAsin: { userId: req.user.id, authorAsin } },
|
||||||
|
});
|
||||||
|
|
||||||
|
if (existing) {
|
||||||
|
return NextResponse.json(
|
||||||
|
{ error: 'AlreadyWatching', message: 'You are already watching this author' },
|
||||||
|
{ status: 409 }
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
const watched = await prisma.watchedAuthor.create({
|
||||||
|
data: {
|
||||||
|
userId: req.user.id,
|
||||||
|
authorAsin,
|
||||||
|
authorName,
|
||||||
|
coverArtUrl: coverArtUrl || null,
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
logger.info(`User ${req.user.id} started watching author "${authorName}" (${authorAsin})`);
|
||||||
|
|
||||||
|
// Trigger immediate targeted check for this author (fire-and-forget)
|
||||||
|
try {
|
||||||
|
const jobQueue = getJobQueueService();
|
||||||
|
await jobQueue.addCheckWatchedItemJob(req.user.id, undefined, authorAsin);
|
||||||
|
logger.info(`Triggered immediate check for watched author "${authorName}" (${authorAsin})`);
|
||||||
|
} catch (error) {
|
||||||
|
logger.error('Failed to trigger immediate watched author check', { error: error instanceof Error ? error.message : String(error) });
|
||||||
|
}
|
||||||
|
|
||||||
|
return NextResponse.json({
|
||||||
|
success: true,
|
||||||
|
author: {
|
||||||
|
id: watched.id,
|
||||||
|
authorAsin: watched.authorAsin,
|
||||||
|
authorName: watched.authorName,
|
||||||
|
coverArtUrl: watched.coverArtUrl,
|
||||||
|
lastCheckedAt: watched.lastCheckedAt,
|
||||||
|
createdAt: watched.createdAt,
|
||||||
|
},
|
||||||
|
}, { status: 201 });
|
||||||
|
} catch (error) {
|
||||||
|
logger.error('Failed to add watched author', { error: error instanceof Error ? error.message : String(error) });
|
||||||
|
|
||||||
|
if (error instanceof z.ZodError) {
|
||||||
|
return NextResponse.json(
|
||||||
|
{ error: 'ValidationError', details: error.errors },
|
||||||
|
{ status: 400 }
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
return NextResponse.json({ error: 'Failed to add watched author' }, { status: 500 });
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
@@ -0,0 +1,52 @@
|
|||||||
|
/**
|
||||||
|
* Component: Watched Series Delete Route
|
||||||
|
* Documentation: documentation/features/watched-lists.md
|
||||||
|
*/
|
||||||
|
|
||||||
|
import { NextRequest, NextResponse } from 'next/server';
|
||||||
|
import { requireAuth, AuthenticatedRequest } from '@/lib/middleware/auth';
|
||||||
|
import { prisma } from '@/lib/db';
|
||||||
|
import { RMABLogger } from '@/lib/utils/logger';
|
||||||
|
|
||||||
|
const logger = RMABLogger.create('API.WatchedSeries');
|
||||||
|
|
||||||
|
/**
|
||||||
|
* DELETE /api/user/watched-series/[id]
|
||||||
|
* Remove a series from the user's watch list (ownership check)
|
||||||
|
*/
|
||||||
|
export async function DELETE(
|
||||||
|
request: NextRequest,
|
||||||
|
{ params }: { params: Promise<{ id: string }> }
|
||||||
|
) {
|
||||||
|
return requireAuth(request, async (req: AuthenticatedRequest) => {
|
||||||
|
try {
|
||||||
|
if (!req.user) {
|
||||||
|
return NextResponse.json({ error: 'Unauthorized' }, { status: 401 });
|
||||||
|
}
|
||||||
|
|
||||||
|
const { id } = await params;
|
||||||
|
|
||||||
|
const watched = await prisma.watchedSeries.findUnique({
|
||||||
|
where: { id },
|
||||||
|
});
|
||||||
|
|
||||||
|
if (!watched) {
|
||||||
|
return NextResponse.json({ error: 'Watched series not found' }, { status: 404 });
|
||||||
|
}
|
||||||
|
|
||||||
|
// Ownership check
|
||||||
|
if (watched.userId !== req.user.id) {
|
||||||
|
return NextResponse.json({ error: 'Forbidden' }, { status: 403 });
|
||||||
|
}
|
||||||
|
|
||||||
|
await prisma.watchedSeries.delete({ where: { id } });
|
||||||
|
|
||||||
|
logger.info(`User ${req.user.id} stopped watching series "${watched.seriesTitle}" (${watched.seriesAsin})`);
|
||||||
|
|
||||||
|
return NextResponse.json({ success: true });
|
||||||
|
} catch (error) {
|
||||||
|
logger.error('Failed to delete watched series', { error: error instanceof Error ? error.message : String(error) });
|
||||||
|
return NextResponse.json({ error: 'Failed to delete watched series' }, { status: 500 });
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
@@ -0,0 +1,125 @@
|
|||||||
|
/**
|
||||||
|
* Component: Watched Series API Routes
|
||||||
|
* Documentation: documentation/features/watched-lists.md
|
||||||
|
*/
|
||||||
|
|
||||||
|
import { NextRequest, NextResponse } from 'next/server';
|
||||||
|
import { requireAuth, AuthenticatedRequest } from '@/lib/middleware/auth';
|
||||||
|
import { prisma } from '@/lib/db';
|
||||||
|
import { getJobQueueService } from '@/lib/services/job-queue.service';
|
||||||
|
import { z } from 'zod';
|
||||||
|
import { RMABLogger } from '@/lib/utils/logger';
|
||||||
|
|
||||||
|
const logger = RMABLogger.create('API.WatchedSeries');
|
||||||
|
|
||||||
|
const AddWatchedSeriesSchema = z.object({
|
||||||
|
seriesAsin: z.string().regex(/^[A-Z0-9]{10}$/, 'Invalid series ASIN'),
|
||||||
|
seriesTitle: z.string().min(1).max(500),
|
||||||
|
coverArtUrl: z.string().url().optional(),
|
||||||
|
});
|
||||||
|
|
||||||
|
/**
|
||||||
|
* GET /api/user/watched-series
|
||||||
|
* List the current user's watched series
|
||||||
|
*/
|
||||||
|
export async function GET(request: NextRequest) {
|
||||||
|
return requireAuth(request, async (req: AuthenticatedRequest) => {
|
||||||
|
try {
|
||||||
|
if (!req.user) {
|
||||||
|
return NextResponse.json({ error: 'Unauthorized' }, { status: 401 });
|
||||||
|
}
|
||||||
|
|
||||||
|
const series = await prisma.watchedSeries.findMany({
|
||||||
|
where: { userId: req.user.id },
|
||||||
|
orderBy: { createdAt: 'desc' },
|
||||||
|
});
|
||||||
|
|
||||||
|
return NextResponse.json({
|
||||||
|
success: true,
|
||||||
|
series: series.map((s) => ({
|
||||||
|
id: s.id,
|
||||||
|
seriesAsin: s.seriesAsin,
|
||||||
|
seriesTitle: s.seriesTitle,
|
||||||
|
coverArtUrl: s.coverArtUrl,
|
||||||
|
lastCheckedAt: s.lastCheckedAt,
|
||||||
|
createdAt: s.createdAt,
|
||||||
|
})),
|
||||||
|
});
|
||||||
|
} catch (error) {
|
||||||
|
logger.error('Failed to list watched series', { error: error instanceof Error ? error.message : String(error) });
|
||||||
|
return NextResponse.json({ error: 'Failed to list watched series' }, { status: 500 });
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* POST /api/user/watched-series
|
||||||
|
* Add a series to the user's watch list
|
||||||
|
*/
|
||||||
|
export async function POST(request: NextRequest) {
|
||||||
|
return requireAuth(request, async (req: AuthenticatedRequest) => {
|
||||||
|
try {
|
||||||
|
if (!req.user) {
|
||||||
|
return NextResponse.json({ error: 'Unauthorized' }, { status: 401 });
|
||||||
|
}
|
||||||
|
|
||||||
|
const body = await req.json();
|
||||||
|
const { seriesAsin, seriesTitle, coverArtUrl } = AddWatchedSeriesSchema.parse(body);
|
||||||
|
|
||||||
|
// Check for duplicate
|
||||||
|
const existing = await prisma.watchedSeries.findUnique({
|
||||||
|
where: { userId_seriesAsin: { userId: req.user.id, seriesAsin } },
|
||||||
|
});
|
||||||
|
|
||||||
|
if (existing) {
|
||||||
|
return NextResponse.json(
|
||||||
|
{ error: 'AlreadyWatching', message: 'You are already watching this series' },
|
||||||
|
{ status: 409 }
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
const watched = await prisma.watchedSeries.create({
|
||||||
|
data: {
|
||||||
|
userId: req.user.id,
|
||||||
|
seriesAsin,
|
||||||
|
seriesTitle,
|
||||||
|
coverArtUrl: coverArtUrl || null,
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
logger.info(`User ${req.user.id} started watching series "${seriesTitle}" (${seriesAsin})`);
|
||||||
|
|
||||||
|
// Trigger immediate targeted check for this series (fire-and-forget)
|
||||||
|
try {
|
||||||
|
const jobQueue = getJobQueueService();
|
||||||
|
await jobQueue.addCheckWatchedItemJob(req.user.id, seriesAsin);
|
||||||
|
logger.info(`Triggered immediate check for watched series "${seriesTitle}" (${seriesAsin})`);
|
||||||
|
} catch (error) {
|
||||||
|
logger.error('Failed to trigger immediate watched series check', { error: error instanceof Error ? error.message : String(error) });
|
||||||
|
}
|
||||||
|
|
||||||
|
return NextResponse.json({
|
||||||
|
success: true,
|
||||||
|
series: {
|
||||||
|
id: watched.id,
|
||||||
|
seriesAsin: watched.seriesAsin,
|
||||||
|
seriesTitle: watched.seriesTitle,
|
||||||
|
coverArtUrl: watched.coverArtUrl,
|
||||||
|
lastCheckedAt: watched.lastCheckedAt,
|
||||||
|
createdAt: watched.createdAt,
|
||||||
|
},
|
||||||
|
}, { status: 201 });
|
||||||
|
} catch (error) {
|
||||||
|
logger.error('Failed to add watched series', { error: error instanceof Error ? error.message : String(error) });
|
||||||
|
|
||||||
|
if (error instanceof z.ZodError) {
|
||||||
|
return NextResponse.json(
|
||||||
|
{ error: 'ValidationError', details: error.errors },
|
||||||
|
{ status: 400 }
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
return NextResponse.json({ error: 'Failed to add watched series' }, { status: 500 });
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
@@ -5,16 +5,17 @@
|
|||||||
|
|
||||||
'use client';
|
'use client';
|
||||||
|
|
||||||
import { use, useCallback } from 'react';
|
import { use, useCallback, useMemo } from 'react';
|
||||||
import { useRouter, useSearchParams } from 'next/navigation';
|
import { useRouter, useSearchParams } from 'next/navigation';
|
||||||
import { Header } from '@/components/layout/Header';
|
import { Header } from '@/components/layout/Header';
|
||||||
import { AudiobookGrid } from '@/components/audiobooks/AudiobookGrid';
|
import { AudiobookGrid } from '@/components/audiobooks/AudiobookGrid';
|
||||||
|
import { LoadMoreBar } from '@/components/ui/LoadMoreBar';
|
||||||
import { AuthorDetailCard, AuthorDetailSkeleton } from '@/components/authors/AuthorDetailCard';
|
import { AuthorDetailCard, AuthorDetailSkeleton } from '@/components/authors/AuthorDetailCard';
|
||||||
import { SimilarAuthorsRow, SimilarAuthorsSkeleton } from '@/components/authors/SimilarAuthorsRow';
|
import { SimilarAuthorsRow, SimilarAuthorsSkeleton } from '@/components/authors/SimilarAuthorsRow';
|
||||||
import { useAuthorDetail, useAuthorBooks } from '@/lib/hooks/useAuthors';
|
import { useAuthorDetail, useAuthorBooks } from '@/lib/hooks/useAuthors';
|
||||||
|
import { Audiobook } from '@/lib/hooks/useAudiobooks';
|
||||||
import { ProtectedRoute } from '@/components/auth/ProtectedRoute';
|
import { ProtectedRoute } from '@/components/auth/ProtectedRoute';
|
||||||
import { CardSizeControls } from '@/components/ui/CardSizeControls';
|
import { SectionToolbar } from '@/components/ui/SectionToolbar';
|
||||||
import { SquareCoversToggle } from '@/components/ui/SquareCoversToggle';
|
|
||||||
import { usePreferences } from '@/contexts/PreferencesContext';
|
import { usePreferences } from '@/contexts/PreferencesContext';
|
||||||
|
|
||||||
export default function AuthorDetailPage({
|
export default function AuthorDetailPage({
|
||||||
@@ -27,11 +28,11 @@ export default function AuthorDetailPage({
|
|||||||
const searchParams = useSearchParams();
|
const searchParams = useSearchParams();
|
||||||
const fromAuthorName = searchParams.get('from');
|
const fromAuthorName = searchParams.get('from');
|
||||||
const { author, isLoading: authorLoading } = useAuthorDetail(asin);
|
const { author, isLoading: authorLoading } = useAuthorDetail(asin);
|
||||||
const { books, totalBooks, isLoading: booksLoading } = useAuthorBooks(
|
const { books, totalBooks, hasMore, isLoading: booksLoading, isLoadingMore, loadMore } = useAuthorBooks(
|
||||||
asin,
|
asin,
|
||||||
author?.name || null
|
author?.name || null
|
||||||
);
|
);
|
||||||
const { cardSize, setCardSize, squareCovers, setSquareCovers } = usePreferences();
|
const { cardSize, setCardSize, squareCovers, setSquareCovers, hideAvailable, setHideAvailable } = usePreferences();
|
||||||
|
|
||||||
const handleBack = useCallback(() => {
|
const handleBack = useCallback(() => {
|
||||||
// Use browser back if we came from within the app, otherwise fallback to /authors
|
// Use browser back if we came from within the app, otherwise fallback to /authors
|
||||||
@@ -42,6 +43,20 @@ export default function AuthorDetailPage({
|
|||||||
}
|
}
|
||||||
}, [router]);
|
}, [router]);
|
||||||
|
|
||||||
|
// Filter out available titles when hideAvailable is enabled
|
||||||
|
const filteredBooks = useMemo(
|
||||||
|
() => hideAvailable ? books.filter((b: Audiobook) => !b.isAvailable && b.requestStatus !== 'completed') : books,
|
||||||
|
[books, hideAvailable]
|
||||||
|
);
|
||||||
|
|
||||||
|
// Header count text: reflects filtered counts
|
||||||
|
const visibleCount = filteredBooks.length;
|
||||||
|
const booksCountText = hasMore && totalBooks > books.length
|
||||||
|
? `${visibleCount.toLocaleString()} of ${totalBooks.toLocaleString()} title${totalBooks !== 1 ? 's' : ''}`
|
||||||
|
: visibleCount > 0
|
||||||
|
? `${visibleCount.toLocaleString()} title${visibleCount !== 1 ? 's' : ''}`
|
||||||
|
: '';
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<ProtectedRoute>
|
<ProtectedRoute>
|
||||||
<div className="min-h-screen">
|
<div className="min-h-screen">
|
||||||
@@ -91,27 +106,42 @@ export default function AuthorDetailPage({
|
|||||||
<h2 className="text-xl sm:text-2xl font-bold text-gray-900 dark:text-gray-100 truncate">
|
<h2 className="text-xl sm:text-2xl font-bold text-gray-900 dark:text-gray-100 truncate">
|
||||||
Books
|
Books
|
||||||
</h2>
|
</h2>
|
||||||
{!booksLoading && totalBooks > 0 && (
|
{!booksLoading && booksCountText && (
|
||||||
<span className="text-sm text-gray-600 dark:text-gray-400 hidden sm:inline whitespace-nowrap">
|
<span className="text-sm text-gray-600 dark:text-gray-400 hidden sm:inline whitespace-nowrap">
|
||||||
({totalBooks} title{totalBooks !== 1 ? 's' : ''})
|
({booksCountText})
|
||||||
</span>
|
</span>
|
||||||
)}
|
)}
|
||||||
<div className="ml-auto flex items-center gap-1">
|
<SectionToolbar
|
||||||
<SquareCoversToggle enabled={squareCovers} onToggle={setSquareCovers} />
|
hideAvailable={hideAvailable}
|
||||||
<CardSizeControls size={cardSize} onSizeChange={setCardSize} />
|
onToggleHideAvailable={setHideAvailable}
|
||||||
</div>
|
squareCovers={squareCovers}
|
||||||
|
onToggleSquareCovers={setSquareCovers}
|
||||||
|
cardSize={cardSize}
|
||||||
|
onCardSizeChange={setCardSize}
|
||||||
|
/>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
{/* Books Grid */}
|
{/* Books Grid */}
|
||||||
<AudiobookGrid
|
<AudiobookGrid
|
||||||
audiobooks={books}
|
audiobooks={filteredBooks}
|
||||||
isLoading={booksLoading}
|
isLoading={booksLoading}
|
||||||
emptyMessage={`No books found for ${author.name}`}
|
emptyMessage={`No books found for ${author.name}`}
|
||||||
cardSize={cardSize}
|
cardSize={cardSize}
|
||||||
squareCovers={squareCovers}
|
squareCovers={squareCovers}
|
||||||
/>
|
/>
|
||||||
|
|
||||||
|
{/* Load More Bar */}
|
||||||
|
{filteredBooks.length > 0 && (
|
||||||
|
<LoadMoreBar
|
||||||
|
loadedCount={filteredBooks.length}
|
||||||
|
totalCount={totalBooks > 0 ? totalBooks : undefined}
|
||||||
|
hasMore={hasMore}
|
||||||
|
isLoading={isLoadingMore}
|
||||||
|
onLoadMore={loadMore}
|
||||||
|
/>
|
||||||
|
)}
|
||||||
</div>
|
</div>
|
||||||
)}
|
)}
|
||||||
</main>
|
</main>
|
||||||
|
|||||||
@@ -300,7 +300,7 @@ export default function BookDatePage() {
|
|||||||
Try Again
|
Try Again
|
||||||
</button>
|
</button>
|
||||||
<button
|
<button
|
||||||
onClick={() => router.push('/settings')}
|
onClick={() => router.push('/admin/settings')}
|
||||||
className="px-6 py-2 border border-gray-300 dark:border-gray-600 text-gray-700 dark:text-gray-300 rounded-lg hover:bg-gray-50 dark:hover:bg-gray-800 transition-colors"
|
className="px-6 py-2 border border-gray-300 dark:border-gray-600 text-gray-700 dark:text-gray-300 rounded-lg hover:bg-gray-50 dark:hover:bg-gray-800 transition-colors"
|
||||||
>
|
>
|
||||||
Go to Settings
|
Go to Settings
|
||||||
@@ -415,6 +415,7 @@ export default function BookDatePage() {
|
|||||||
isAvailable={currentRec.isAvailable}
|
isAvailable={currentRec.isAvailable}
|
||||||
requestedByUsername={currentRec.requestedByUsername}
|
requestedByUsername={currentRec.requestedByUsername}
|
||||||
hideRequestActions
|
hideRequestActions
|
||||||
|
aiReason={currentRec.aiReason}
|
||||||
/>
|
/>
|
||||||
) : null;
|
) : null;
|
||||||
})()}
|
})()}
|
||||||
|
|||||||
@@ -197,6 +197,31 @@ body {
|
|||||||
animation: toast-slide-in 0.3s ease-out;
|
animation: toast-slide-in 0.3s ease-out;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/* Confirmation Dialog */
|
||||||
|
@keyframes dialog-backdrop-in {
|
||||||
|
from { opacity: 0; }
|
||||||
|
to { opacity: 1; }
|
||||||
|
}
|
||||||
|
|
||||||
|
@keyframes dialog-panel-in {
|
||||||
|
from {
|
||||||
|
opacity: 0;
|
||||||
|
transform: scale(0.95) translateY(8px);
|
||||||
|
}
|
||||||
|
to {
|
||||||
|
opacity: 1;
|
||||||
|
transform: scale(1) translateY(0);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
.animate-dialog-backdrop {
|
||||||
|
animation: dialog-backdrop-in 0.15s ease-out forwards;
|
||||||
|
}
|
||||||
|
|
||||||
|
.animate-dialog-panel {
|
||||||
|
animation: dialog-panel-in 0.2s cubic-bezier(0.16, 1, 0.3, 1) forwards;
|
||||||
|
}
|
||||||
|
|
||||||
/* Hide scrollbar while keeping scroll functional */
|
/* Hide scrollbar while keeping scroll functional */
|
||||||
.scrollbar-hide {
|
.scrollbar-hide {
|
||||||
-ms-overflow-style: none;
|
-ms-overflow-style: none;
|
||||||
|
|||||||
@@ -486,6 +486,7 @@ function LoginContent() {
|
|||||||
quality={70}
|
quality={70}
|
||||||
priority={index < 10}
|
priority={index < 10}
|
||||||
loading={index < 10 ? 'eager' : 'lazy'}
|
loading={index < 10 ? 'eager' : 'lazy'}
|
||||||
|
onError={(e) => { (e.target as HTMLImageElement).src = '/placeholder_cover.svg'; }}
|
||||||
/>
|
/>
|
||||||
<div className="absolute inset-0 bg-gradient-to-t from-black/40 to-transparent" />
|
<div className="absolute inset-0 bg-gradient-to-t from-black/40 to-transparent" />
|
||||||
</div>
|
</div>
|
||||||
|
|||||||
+160
-157
@@ -1,186 +1,189 @@
|
|||||||
/**
|
/**
|
||||||
* Component: Homepage - Audiobook Discovery
|
* Component: Homepage - Audiobook Discovery (Dynamic Sections)
|
||||||
* Documentation: documentation/frontend/components.md
|
* Documentation: documentation/features/home-sections.md
|
||||||
*/
|
*/
|
||||||
|
|
||||||
'use client';
|
'use client';
|
||||||
|
|
||||||
import { useState, useRef } from 'react';
|
import { useState, useRef, useEffect, useCallback, createRef } from 'react';
|
||||||
import { Header } from '@/components/layout/Header';
|
import { Header } from '@/components/layout/Header';
|
||||||
import { AudiobookGrid } from '@/components/audiobooks/AudiobookGrid';
|
|
||||||
import { useAudiobooks } from '@/lib/hooks/useAudiobooks';
|
|
||||||
import { ProtectedRoute } from '@/components/auth/ProtectedRoute';
|
import { ProtectedRoute } from '@/components/auth/ProtectedRoute';
|
||||||
import { StickyPagination } from '@/components/ui/StickyPagination';
|
import { UnifiedPagination, PaginationSection } from '@/components/ui/UnifiedPagination';
|
||||||
import { CardSizeControls } from '@/components/ui/CardSizeControls';
|
import { HomeSection, SECTION_DOT_COLORS } from '@/components/home/HomeSection';
|
||||||
import { SquareCoversToggle } from '@/components/ui/SquareCoversToggle';
|
import { HomeSectionConfigModal } from '@/components/home/HomeSectionConfigModal';
|
||||||
|
import { useHomeSections } from '@/lib/hooks/useHomeSections';
|
||||||
import { usePreferences } from '@/contexts/PreferencesContext';
|
import { usePreferences } from '@/contexts/PreferencesContext';
|
||||||
|
import { Cog6ToothIcon } from '@heroicons/react/24/outline';
|
||||||
|
|
||||||
|
function getSectionTitle(sectionType: string, categoryName?: string | null): string {
|
||||||
|
if (sectionType === 'popular') return 'Popular Audiobooks';
|
||||||
|
if (sectionType === 'new_releases') return 'New Releases';
|
||||||
|
return categoryName || 'Category';
|
||||||
|
}
|
||||||
|
|
||||||
export default function HomePage() {
|
export default function HomePage() {
|
||||||
const [popularPage, setPopularPage] = useState(1);
|
const { sections, nextRefresh, isLoading: sectionsLoading, saveSections } = useHomeSections();
|
||||||
const [newReleasesPage, setNewReleasesPage] = useState(1);
|
const { cardSize, setCardSize, squareCovers, setSquareCovers, hideAvailable, setHideAvailable } = usePreferences();
|
||||||
const { cardSize, setCardSize, squareCovers, setSquareCovers } = usePreferences();
|
|
||||||
|
// Per-section pagination state
|
||||||
|
const [pages, setPages] = useState<Record<string, number>>({});
|
||||||
|
const [totalPagesMap, setTotalPagesMap] = useState<Record<string, number>>({});
|
||||||
|
const [configOpen, setConfigOpen] = useState(false);
|
||||||
|
|
||||||
// Refs for auto-scrolling to section tops
|
|
||||||
const popularSectionRef = useRef<HTMLElement>(null);
|
|
||||||
const newReleasesSectionRef = useRef<HTMLElement>(null);
|
|
||||||
const footerRef = useRef<HTMLElement>(null);
|
const footerRef = useRef<HTMLElement>(null);
|
||||||
|
|
||||||
const {
|
// Create stable refs for each section
|
||||||
audiobooks: popular,
|
const sectionRefsMap = useRef<Map<string, React.RefObject<HTMLElement | null>>>(new Map());
|
||||||
isLoading: loadingPopular,
|
|
||||||
totalPages: popularTotalPages,
|
|
||||||
message: popularMessage,
|
|
||||||
} = useAudiobooks('popular', 20, popularPage);
|
|
||||||
|
|
||||||
const {
|
const getSectionKey = (s: { sectionType: string; categoryId: string | null }) =>
|
||||||
audiobooks: newReleases,
|
s.sectionType === 'category' ? `category:${s.categoryId}` : s.sectionType;
|
||||||
isLoading: loadingNewReleases,
|
|
||||||
totalPages: newReleasesTotalPages,
|
|
||||||
message: newReleasesMessage,
|
|
||||||
} = useAudiobooks('new-releases', 20, newReleasesPage);
|
|
||||||
|
|
||||||
// Handle page changes with auto-scroll to section top
|
// Ensure refs exist for current sections
|
||||||
const handlePopularPageChange = (page: number) => {
|
sections.forEach((s) => {
|
||||||
setPopularPage(page);
|
const key = getSectionKey(s);
|
||||||
popularSectionRef.current?.scrollIntoView({ behavior: 'smooth', block: 'start' });
|
if (!sectionRefsMap.current.has(key)) {
|
||||||
};
|
sectionRefsMap.current.set(key, createRef<HTMLElement>());
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
const handleNewReleasesPageChange = (page: number) => {
|
// Reset pages and totalPages when hideAvailable changes
|
||||||
setNewReleasesPage(page);
|
useEffect(() => {
|
||||||
newReleasesSectionRef.current?.scrollIntoView({ behavior: 'smooth', block: 'start' });
|
setPages({});
|
||||||
};
|
setTotalPagesMap({});
|
||||||
|
}, [hideAvailable]);
|
||||||
|
|
||||||
|
const getPage = (key: string) => pages[key] || 1;
|
||||||
|
const setPage = useCallback((key: string, page: number) => {
|
||||||
|
setPages((prev) => ({ ...prev, [key]: page }));
|
||||||
|
}, []);
|
||||||
|
const handleTotalPagesChange = useCallback((key: string, totalPages: number) => {
|
||||||
|
setTotalPagesMap((prev) => {
|
||||||
|
if (prev[key] === totalPages) return prev;
|
||||||
|
return { ...prev, [key]: totalPages };
|
||||||
|
});
|
||||||
|
}, []);
|
||||||
|
|
||||||
|
// Build pagination sections for the floating pill
|
||||||
|
const paginationSections: PaginationSection[] = sections.map((s, i) => {
|
||||||
|
const key = getSectionKey(s);
|
||||||
|
const ref = sectionRefsMap.current.get(key)!;
|
||||||
|
return {
|
||||||
|
label: getSectionTitle(s.sectionType, s.categoryName),
|
||||||
|
accentColor: SECTION_DOT_COLORS[i % SECTION_DOT_COLORS.length],
|
||||||
|
currentPage: getPage(key),
|
||||||
|
totalPages: totalPagesMap[key] || 1,
|
||||||
|
onPageChange: (page: number) => {
|
||||||
|
setPage(key, page);
|
||||||
|
ref.current?.scrollIntoView({ behavior: 'smooth', block: 'start' });
|
||||||
|
},
|
||||||
|
sectionRef: ref,
|
||||||
|
onScrollToSection: () =>
|
||||||
|
ref.current?.scrollIntoView({ behavior: 'smooth', block: 'start' }),
|
||||||
|
};
|
||||||
|
});
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<ProtectedRoute>
|
<ProtectedRoute>
|
||||||
<div className="min-h-screen">
|
<div className="min-h-screen">
|
||||||
<Header />
|
<Header />
|
||||||
|
|
||||||
<main className="container mx-auto px-4 py-6 sm:py-8 max-w-7xl space-y-8 sm:space-y-12">
|
<main className="container mx-auto px-4 py-6 sm:py-8 max-w-7xl space-y-8 sm:space-y-12">
|
||||||
{/* Popular Audiobooks Section */}
|
{/* Loading state */}
|
||||||
<section ref={popularSectionRef} className="relative">
|
{sectionsLoading && (
|
||||||
{/* Sticky Section Header */}
|
<div className="flex justify-center py-20">
|
||||||
<div className="sticky top-14 sm:top-16 z-30 mb-4 sm:mb-6">
|
<div className="animate-spin h-8 w-8 border-2 border-blue-500 border-t-transparent rounded-full" />
|
||||||
<div className="bg-white/90 dark:bg-gray-800/90 backdrop-blur-md rounded-2xl px-4 sm:px-6 py-3 border border-gray-200/50 dark:border-gray-700/50 shadow-sm">
|
</div>
|
||||||
<div className="flex items-center gap-3">
|
)}
|
||||||
<div className="w-1 h-6 bg-gradient-to-b from-blue-500 to-purple-500 rounded-full" />
|
|
||||||
<h2 className="text-xl sm:text-2xl md:text-3xl font-bold text-gray-900 dark:text-gray-100 truncate">
|
{/* Empty state */}
|
||||||
Popular Audiobooks
|
{!sectionsLoading && sections.length === 0 && (
|
||||||
</h2>
|
<div className="text-center py-20">
|
||||||
<div className="ml-auto flex items-center gap-1">
|
<p className="text-gray-500 dark:text-gray-400 mb-4">
|
||||||
<SquareCoversToggle enabled={squareCovers} onToggle={setSquareCovers} />
|
No sections configured. Click Customize to add sections to your home page.
|
||||||
<CardSizeControls size={cardSize} onSizeChange={setCardSize} />
|
</p>
|
||||||
</div>
|
<button
|
||||||
</div>
|
onClick={() => setConfigOpen(true)}
|
||||||
|
className="inline-flex items-center px-4 py-2 bg-blue-600 text-white font-medium rounded-lg hover:bg-blue-700 transition-colors"
|
||||||
|
>
|
||||||
|
<Cog6ToothIcon className="w-4 h-4 mr-2" />
|
||||||
|
Customize Home
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
|
||||||
|
{/* Dynamic sections */}
|
||||||
|
{!sectionsLoading &&
|
||||||
|
sections.map((section, index) => {
|
||||||
|
const key = getSectionKey(section);
|
||||||
|
const ref = sectionRefsMap.current.get(key)!;
|
||||||
|
|
||||||
|
return (
|
||||||
|
<HomeSection
|
||||||
|
key={key}
|
||||||
|
sectionType={section.sectionType as 'popular' | 'new_releases' | 'category'}
|
||||||
|
categoryId={section.categoryId}
|
||||||
|
categoryName={section.categoryName}
|
||||||
|
colorIndex={index}
|
||||||
|
page={getPage(key)}
|
||||||
|
onPageChange={(page) => {
|
||||||
|
setPage(key, page);
|
||||||
|
ref.current?.scrollIntoView({ behavior: 'smooth', block: 'start' });
|
||||||
|
}}
|
||||||
|
sectionRef={ref}
|
||||||
|
cardSize={cardSize}
|
||||||
|
squareCovers={squareCovers}
|
||||||
|
hideAvailable={hideAvailable}
|
||||||
|
onToggleHideAvailable={setHideAvailable}
|
||||||
|
onToggleSquareCovers={setSquareCovers}
|
||||||
|
onCardSizeChange={setCardSize}
|
||||||
|
onConfigOpen={index === 0 ? () => setConfigOpen(true) : undefined}
|
||||||
|
onTotalPagesChange={(tp) => handleTotalPagesChange(key, tp)}
|
||||||
|
nextRefresh={nextRefresh}
|
||||||
|
/>
|
||||||
|
);
|
||||||
|
})}
|
||||||
|
|
||||||
|
{/* Call to Action */}
|
||||||
|
<section className="bg-gradient-to-br from-blue-50 to-indigo-50 dark:from-blue-900/20 dark:to-indigo-900/20 rounded-2xl p-6 sm:p-8 text-center border border-blue-200/50 dark:border-blue-800/50 shadow-sm">
|
||||||
|
<h3 className="text-xl sm:text-2xl font-bold text-gray-900 dark:text-gray-100 mb-2">
|
||||||
|
Can't find what you're looking for?
|
||||||
|
</h3>
|
||||||
|
<p className="text-gray-600 dark:text-gray-400 mb-4">
|
||||||
|
Use our search to find any audiobook from Audible
|
||||||
|
</p>
|
||||||
|
<a
|
||||||
|
href="/search"
|
||||||
|
className="inline-flex items-center px-6 py-3 bg-blue-600 text-white font-medium rounded-lg hover:bg-blue-700 transition-colors shadow-md hover:shadow-lg"
|
||||||
|
>
|
||||||
|
Search Audiobooks
|
||||||
|
</a>
|
||||||
|
</section>
|
||||||
|
</main>
|
||||||
|
|
||||||
|
{/* Footer */}
|
||||||
|
<footer ref={footerRef} className="bg-white dark:bg-gray-800 border-t border-gray-200 dark:border-gray-700 mt-16">
|
||||||
|
<div className="container mx-auto px-4 py-6 max-w-7xl">
|
||||||
|
<div className="text-center text-sm text-gray-600 dark:text-gray-400">
|
||||||
|
<p>ReadMeABook - Audiobook Library Management System</p>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
</footer>
|
||||||
|
|
||||||
{/* Section Content */}
|
{/* Unified Pagination — dynamic sections */}
|
||||||
<div className="bg-white/40 dark:bg-gray-800/40 backdrop-blur-sm rounded-2xl p-4 sm:p-6 border border-gray-200/50 dark:border-gray-700/50 shadow-sm">
|
{paginationSections.length > 0 && (
|
||||||
{popularMessage && !loadingPopular && popular.length === 0 ? (
|
<UnifiedPagination
|
||||||
<div className="bg-yellow-50 dark:bg-yellow-900/20 border border-yellow-200 dark:border-yellow-800 rounded-lg p-6 text-center">
|
footerRef={footerRef}
|
||||||
<p className="text-yellow-800 dark:text-yellow-200 mb-2 font-medium">
|
sections={paginationSections}
|
||||||
No popular audiobooks found
|
/>
|
||||||
</p>
|
)}
|
||||||
<p className="text-yellow-700 dark:text-yellow-300 text-sm">
|
|
||||||
{popularMessage}
|
|
||||||
</p>
|
|
||||||
</div>
|
|
||||||
) : (
|
|
||||||
<AudiobookGrid
|
|
||||||
audiobooks={popular}
|
|
||||||
isLoading={loadingPopular}
|
|
||||||
emptyMessage="No popular audiobooks available"
|
|
||||||
cardSize={cardSize}
|
|
||||||
squareCovers={squareCovers}
|
|
||||||
/>
|
|
||||||
)}
|
|
||||||
</div>
|
|
||||||
</section>
|
|
||||||
|
|
||||||
{/* New Releases Section */}
|
{/* Config Modal */}
|
||||||
<section ref={newReleasesSectionRef} className="relative">
|
<HomeSectionConfigModal
|
||||||
{/* Sticky Section Header */}
|
isOpen={configOpen}
|
||||||
<div className="sticky top-14 sm:top-16 z-30 mb-4 sm:mb-6">
|
onClose={() => setConfigOpen(false)}
|
||||||
<div className="bg-white/90 dark:bg-gray-800/90 backdrop-blur-md rounded-2xl px-4 sm:px-6 py-3 border border-gray-200/50 dark:border-gray-700/50 shadow-sm">
|
sections={sections}
|
||||||
<div className="flex items-center gap-3">
|
onSave={saveSections}
|
||||||
<div className="w-1 h-6 bg-gradient-to-b from-emerald-500 to-teal-500 rounded-full" />
|
/>
|
||||||
<h2 className="text-xl sm:text-2xl md:text-3xl font-bold text-gray-900 dark:text-gray-100 truncate">
|
|
||||||
New Releases
|
|
||||||
</h2>
|
|
||||||
<div className="ml-auto flex items-center gap-1">
|
|
||||||
<SquareCoversToggle enabled={squareCovers} onToggle={setSquareCovers} />
|
|
||||||
<CardSizeControls size={cardSize} onSizeChange={setCardSize} />
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
{/* Section Content */}
|
|
||||||
<div className="bg-white/40 dark:bg-gray-800/40 backdrop-blur-sm rounded-2xl p-4 sm:p-6 border border-gray-200/50 dark:border-gray-700/50 shadow-sm">
|
|
||||||
{newReleasesMessage && !loadingNewReleases && newReleases.length === 0 ? (
|
|
||||||
<div className="bg-yellow-50 dark:bg-yellow-900/20 border border-yellow-200 dark:border-yellow-800 rounded-lg p-6 text-center">
|
|
||||||
<p className="text-yellow-800 dark:text-yellow-200 mb-2 font-medium">
|
|
||||||
No new releases found
|
|
||||||
</p>
|
|
||||||
<p className="text-yellow-700 dark:text-yellow-300 text-sm">
|
|
||||||
{newReleasesMessage}
|
|
||||||
</p>
|
|
||||||
</div>
|
|
||||||
) : (
|
|
||||||
<AudiobookGrid
|
|
||||||
audiobooks={newReleases}
|
|
||||||
isLoading={loadingNewReleases}
|
|
||||||
emptyMessage="No new releases available"
|
|
||||||
cardSize={cardSize}
|
|
||||||
squareCovers={squareCovers}
|
|
||||||
/>
|
|
||||||
)}
|
|
||||||
</div>
|
|
||||||
</section>
|
|
||||||
|
|
||||||
{/* Call to Action */}
|
|
||||||
<section className="bg-gradient-to-br from-blue-50 to-indigo-50 dark:from-blue-900/20 dark:to-indigo-900/20 rounded-2xl p-6 sm:p-8 text-center border border-blue-200/50 dark:border-blue-800/50 shadow-sm">
|
|
||||||
<h3 className="text-xl sm:text-2xl font-bold text-gray-900 dark:text-gray-100 mb-2">
|
|
||||||
Can't find what you're looking for?
|
|
||||||
</h3>
|
|
||||||
<p className="text-gray-600 dark:text-gray-400 mb-4">
|
|
||||||
Use our search to find any audiobook from Audible
|
|
||||||
</p>
|
|
||||||
<a
|
|
||||||
href="/search"
|
|
||||||
className="inline-flex items-center px-6 py-3 bg-blue-600 text-white font-medium rounded-lg hover:bg-blue-700 transition-colors shadow-md hover:shadow-lg"
|
|
||||||
>
|
|
||||||
Search Audiobooks
|
|
||||||
</a>
|
|
||||||
</section>
|
|
||||||
</main>
|
|
||||||
|
|
||||||
{/* Footer */}
|
|
||||||
<footer ref={footerRef} className="bg-white dark:bg-gray-800 border-t border-gray-200 dark:border-gray-700 mt-16">
|
|
||||||
<div className="container mx-auto px-4 py-6 max-w-7xl">
|
|
||||||
<div className="text-center text-sm text-gray-600 dark:text-gray-400">
|
|
||||||
<p>ReadMeABook - Audiobook Library Management System</p>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</footer>
|
|
||||||
|
|
||||||
{/* Sticky Pagination Controls */}
|
|
||||||
<StickyPagination
|
|
||||||
currentPage={popularPage}
|
|
||||||
totalPages={popularTotalPages}
|
|
||||||
onPageChange={handlePopularPageChange}
|
|
||||||
sectionRef={popularSectionRef}
|
|
||||||
footerRef={footerRef}
|
|
||||||
label="Popular Audiobooks"
|
|
||||||
/>
|
|
||||||
<StickyPagination
|
|
||||||
currentPage={newReleasesPage}
|
|
||||||
totalPages={newReleasesTotalPages}
|
|
||||||
onPageChange={handleNewReleasesPageChange}
|
|
||||||
sectionRef={newReleasesSectionRef}
|
|
||||||
footerRef={footerRef}
|
|
||||||
label="New Releases"
|
|
||||||
/>
|
|
||||||
</div>
|
</div>
|
||||||
</ProtectedRoute>
|
</ProtectedRoute>
|
||||||
);
|
);
|
||||||
|
|||||||
@@ -11,7 +11,9 @@ import { RequestCard } from '@/components/requests/RequestCard';
|
|||||||
import { useAuth } from '@/contexts/AuthContext';
|
import { useAuth } from '@/contexts/AuthContext';
|
||||||
import { useRequests } from '@/lib/hooks/useRequests';
|
import { useRequests } from '@/lib/hooks/useRequests';
|
||||||
import { cn } from '@/lib/utils/cn';
|
import { cn } from '@/lib/utils/cn';
|
||||||
import { GoodreadsShelvesSection } from '@/components/profile/GoodreadsShelvesSection';
|
import { ShelvesSection } from '@/components/profile/ShelvesSection';
|
||||||
|
import { ApiTokensSection } from '@/components/profile/ApiTokensSection';
|
||||||
|
import { WatchedSeriesSection, WatchedAuthorsSection } from '@/components/profile/WatchedListsSection';
|
||||||
|
|
||||||
const statConfig = [
|
const statConfig = [
|
||||||
{ key: 'total', label: 'Total', color: 'text-gray-900 dark:text-white' },
|
{ key: 'total', label: 'Total', color: 'text-gray-900 dark:text-white' },
|
||||||
@@ -139,8 +141,14 @@ export default function ProfilePage() {
|
|||||||
</div>
|
</div>
|
||||||
</section>
|
</section>
|
||||||
|
|
||||||
{/* Goodreads Shelves */}
|
{/* Generic Shelves Section */}
|
||||||
<GoodreadsShelvesSection />
|
<ShelvesSection />
|
||||||
|
|
||||||
|
{/* Watched Series */}
|
||||||
|
<WatchedSeriesSection />
|
||||||
|
|
||||||
|
{/* Watched Authors */}
|
||||||
|
<WatchedAuthorsSection />
|
||||||
|
|
||||||
{/* Active Downloads */}
|
{/* Active Downloads */}
|
||||||
{activeDownloads.length > 0 && (
|
{activeDownloads.length > 0 && (
|
||||||
@@ -233,6 +241,9 @@ export default function ProfilePage() {
|
|||||||
</div>
|
</div>
|
||||||
)}
|
)}
|
||||||
</section>
|
</section>
|
||||||
|
|
||||||
|
{/* API Tokens */}
|
||||||
|
<ApiTokensSection />
|
||||||
</main>
|
</main>
|
||||||
</div>
|
</div>
|
||||||
);
|
);
|
||||||
|
|||||||
+41
-37
@@ -5,41 +5,48 @@
|
|||||||
|
|
||||||
'use client';
|
'use client';
|
||||||
|
|
||||||
import { useState, useEffect, useCallback } from 'react';
|
import { useState, useEffect, useCallback, useMemo } from 'react';
|
||||||
import { Header } from '@/components/layout/Header';
|
import { Header } from '@/components/layout/Header';
|
||||||
import { AudiobookGrid } from '@/components/audiobooks/AudiobookGrid';
|
import { AudiobookGrid } from '@/components/audiobooks/AudiobookGrid';
|
||||||
import { useSearch } from '@/lib/hooks/useAudiobooks';
|
import { LoadMoreBar } from '@/components/ui/LoadMoreBar';
|
||||||
|
import { useSearch, Audiobook } from '@/lib/hooks/useAudiobooks';
|
||||||
import { ProtectedRoute } from '@/components/auth/ProtectedRoute';
|
import { ProtectedRoute } from '@/components/auth/ProtectedRoute';
|
||||||
import { CardSizeControls } from '@/components/ui/CardSizeControls';
|
import { SectionToolbar } from '@/components/ui/SectionToolbar';
|
||||||
import { SquareCoversToggle } from '@/components/ui/SquareCoversToggle';
|
|
||||||
import { usePreferences } from '@/contexts/PreferencesContext';
|
import { usePreferences } from '@/contexts/PreferencesContext';
|
||||||
|
|
||||||
export default function SearchPage() {
|
export default function SearchPage() {
|
||||||
const [query, setQuery] = useState('');
|
const [query, setQuery] = useState('');
|
||||||
const [debouncedQuery, setDebouncedQuery] = useState('');
|
const [debouncedQuery, setDebouncedQuery] = useState('');
|
||||||
const [page, setPage] = useState(1);
|
const { cardSize, setCardSize, squareCovers, setSquareCovers, hideAvailable, setHideAvailable } = usePreferences();
|
||||||
const { cardSize, setCardSize, squareCovers, setSquareCovers } = usePreferences();
|
|
||||||
|
|
||||||
// Debounce search query
|
// Debounce search query
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
const timer = setTimeout(() => {
|
const timer = setTimeout(() => {
|
||||||
setDebouncedQuery(query);
|
setDebouncedQuery(query);
|
||||||
setPage(1); // Reset to first page on new search
|
|
||||||
}, 500);
|
}, 500);
|
||||||
|
|
||||||
return () => clearTimeout(timer);
|
return () => clearTimeout(timer);
|
||||||
}, [query]);
|
}, [query]);
|
||||||
|
|
||||||
const { results, totalResults, hasMore, isLoading } = useSearch(debouncedQuery, page);
|
const { results, totalResults, hasMore, isLoading, isLoadingMore, loadMore } = useSearch(debouncedQuery);
|
||||||
|
|
||||||
|
// Filter out available titles when hideAvailable is enabled
|
||||||
|
const filteredResults = useMemo(
|
||||||
|
() => hideAvailable ? results.filter((b: Audiobook) => !b.isAvailable && b.requestStatus !== 'completed') : results,
|
||||||
|
[results, hideAvailable]
|
||||||
|
);
|
||||||
|
|
||||||
const handleSearch = useCallback((e: React.FormEvent) => {
|
const handleSearch = useCallback((e: React.FormEvent) => {
|
||||||
e.preventDefault();
|
e.preventDefault();
|
||||||
setPage(1);
|
|
||||||
}, []);
|
}, []);
|
||||||
|
|
||||||
const handleLoadMore = useCallback(() => {
|
// Header count text: reflects filtered counts
|
||||||
setPage((prev) => prev + 1);
|
const visibleCount = filteredResults.length;
|
||||||
}, []);
|
const countText = hasMore && totalResults > 0
|
||||||
|
? `${visibleCount.toLocaleString()} of ${totalResults.toLocaleString()} result${totalResults !== 1 ? 's' : ''}`
|
||||||
|
: visibleCount > 0
|
||||||
|
? `${visibleCount.toLocaleString()} result${visibleCount !== 1 ? 's' : ''}`
|
||||||
|
: '';
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<ProtectedRoute>
|
<ProtectedRoute>
|
||||||
@@ -113,45 +120,42 @@ export default function SearchPage() {
|
|||||||
<h2 className="text-xl sm:text-2xl font-bold text-gray-900 dark:text-gray-100 truncate">
|
<h2 className="text-xl sm:text-2xl font-bold text-gray-900 dark:text-gray-100 truncate">
|
||||||
Search Results
|
Search Results
|
||||||
</h2>
|
</h2>
|
||||||
{!isLoading && totalResults > 0 && (
|
{!isLoading && countText && (
|
||||||
<span className="text-sm text-gray-600 dark:text-gray-400 hidden sm:inline whitespace-nowrap">
|
<span className="text-sm text-gray-600 dark:text-gray-400 hidden sm:inline whitespace-nowrap">
|
||||||
({totalResults.toLocaleString()} result{totalResults !== 1 ? 's' : ''})
|
({countText})
|
||||||
</span>
|
</span>
|
||||||
)}
|
)}
|
||||||
<div className="ml-auto flex items-center gap-1">
|
<SectionToolbar
|
||||||
<SquareCoversToggle enabled={squareCovers} onToggle={setSquareCovers} />
|
hideAvailable={hideAvailable}
|
||||||
<CardSizeControls size={cardSize} onSizeChange={setCardSize} />
|
onToggleHideAvailable={setHideAvailable}
|
||||||
</div>
|
squareCovers={squareCovers}
|
||||||
|
onToggleSquareCovers={setSquareCovers}
|
||||||
|
cardSize={cardSize}
|
||||||
|
onCardSizeChange={setCardSize}
|
||||||
|
/>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
{/* Results Grid */}
|
{/* Results Grid */}
|
||||||
<AudiobookGrid
|
<AudiobookGrid
|
||||||
audiobooks={results}
|
audiobooks={filteredResults}
|
||||||
isLoading={!!(isLoading && page === 1)}
|
isLoading={isLoading}
|
||||||
emptyMessage={`No results found for "${debouncedQuery}"`}
|
emptyMessage={`No results found for "${debouncedQuery}"`}
|
||||||
cardSize={cardSize}
|
cardSize={cardSize}
|
||||||
squareCovers={squareCovers}
|
squareCovers={squareCovers}
|
||||||
/>
|
/>
|
||||||
|
|
||||||
{/* Load More */}
|
{/* Load More Bar */}
|
||||||
{hasMore && !isLoading && (
|
{filteredResults.length > 0 && (
|
||||||
<div className="flex justify-center">
|
<LoadMoreBar
|
||||||
<button
|
loadedCount={filteredResults.length}
|
||||||
onClick={handleLoadMore}
|
totalCount={totalResults}
|
||||||
className="px-6 py-3 bg-blue-600 text-white font-medium rounded-lg hover:bg-blue-700 transition-colors"
|
hasMore={hasMore}
|
||||||
>
|
isLoading={isLoadingMore}
|
||||||
Load More Results
|
onLoadMore={loadMore}
|
||||||
</button>
|
itemLabel="results"
|
||||||
</div>
|
/>
|
||||||
)}
|
|
||||||
|
|
||||||
{/* Loading More Indicator */}
|
|
||||||
{isLoading && page > 1 && (
|
|
||||||
<div className="flex justify-center">
|
|
||||||
<div className="animate-spin rounded-full h-8 w-8 border-b-2 border-blue-600"></div>
|
|
||||||
</div>
|
|
||||||
)}
|
)}
|
||||||
</div>
|
</div>
|
||||||
) : (
|
) : (
|
||||||
|
|||||||
@@ -5,16 +5,17 @@
|
|||||||
|
|
||||||
'use client';
|
'use client';
|
||||||
|
|
||||||
import { use, useCallback } from 'react';
|
import { use, useCallback, useMemo } from 'react';
|
||||||
import { useRouter, useSearchParams } from 'next/navigation';
|
import { useRouter, useSearchParams } from 'next/navigation';
|
||||||
import { Header } from '@/components/layout/Header';
|
import { Header } from '@/components/layout/Header';
|
||||||
import { AudiobookGrid } from '@/components/audiobooks/AudiobookGrid';
|
import { AudiobookGrid } from '@/components/audiobooks/AudiobookGrid';
|
||||||
|
import { LoadMoreBar } from '@/components/ui/LoadMoreBar';
|
||||||
import { SeriesDetailCard, SeriesDetailSkeleton } from '@/components/series/SeriesDetailCard';
|
import { SeriesDetailCard, SeriesDetailSkeleton } from '@/components/series/SeriesDetailCard';
|
||||||
import { SimilarSeriesRow, SimilarSeriesSkeleton } from '@/components/series/SimilarSeriesRow';
|
import { SimilarSeriesRow, SimilarSeriesSkeleton } from '@/components/series/SimilarSeriesRow';
|
||||||
import { useSeriesDetail } from '@/lib/hooks/useSeries';
|
import { useSeriesDetail } from '@/lib/hooks/useSeries';
|
||||||
|
import { Audiobook } from '@/lib/hooks/useAudiobooks';
|
||||||
import { ProtectedRoute } from '@/components/auth/ProtectedRoute';
|
import { ProtectedRoute } from '@/components/auth/ProtectedRoute';
|
||||||
import { CardSizeControls } from '@/components/ui/CardSizeControls';
|
import { SectionToolbar } from '@/components/ui/SectionToolbar';
|
||||||
import { SquareCoversToggle } from '@/components/ui/SquareCoversToggle';
|
|
||||||
import { usePreferences } from '@/contexts/PreferencesContext';
|
import { usePreferences } from '@/contexts/PreferencesContext';
|
||||||
|
|
||||||
export default function SeriesDetailPage({
|
export default function SeriesDetailPage({
|
||||||
@@ -26,8 +27,8 @@ export default function SeriesDetailPage({
|
|||||||
const router = useRouter();
|
const router = useRouter();
|
||||||
const searchParams = useSearchParams();
|
const searchParams = useSearchParams();
|
||||||
const fromSeriesTitle = searchParams.get('from');
|
const fromSeriesTitle = searchParams.get('from');
|
||||||
const { series, isLoading: seriesLoading } = useSeriesDetail(asin);
|
const { series, hasMore, isLoading: seriesLoading, isLoadingMore, loadMore } = useSeriesDetail(asin);
|
||||||
const { cardSize, setCardSize, squareCovers, setSquareCovers } = usePreferences();
|
const { cardSize, setCardSize, squareCovers, setSquareCovers, hideAvailable, setHideAvailable } = usePreferences();
|
||||||
|
|
||||||
const handleBack = useCallback(() => {
|
const handleBack = useCallback(() => {
|
||||||
// Use browser back if we came from within the app, otherwise fallback to /series
|
// Use browser back if we came from within the app, otherwise fallback to /series
|
||||||
@@ -38,6 +39,24 @@ export default function SeriesDetailPage({
|
|||||||
}
|
}
|
||||||
}, [router]);
|
}, [router]);
|
||||||
|
|
||||||
|
// Filter out available titles when hideAvailable is enabled
|
||||||
|
const filteredBooks = useMemo(
|
||||||
|
() => series && hideAvailable
|
||||||
|
? series.books.filter((b: Audiobook) => !b.isAvailable && b.requestStatus !== 'completed')
|
||||||
|
: series?.books ?? [],
|
||||||
|
[series, hideAvailable]
|
||||||
|
);
|
||||||
|
|
||||||
|
// Header count text: reflects filtered counts
|
||||||
|
const visibleCount = filteredBooks.length;
|
||||||
|
const booksCountText = series
|
||||||
|
? hasMore && series.bookCount > series.books.length
|
||||||
|
? `${visibleCount.toLocaleString()} of ${series.bookCount.toLocaleString()} title${series.bookCount !== 1 ? 's' : ''}`
|
||||||
|
: visibleCount > 0
|
||||||
|
? `${visibleCount.toLocaleString()} title${visibleCount !== 1 ? 's' : ''}`
|
||||||
|
: ''
|
||||||
|
: '';
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<ProtectedRoute>
|
<ProtectedRoute>
|
||||||
<div className="min-h-screen">
|
<div className="min-h-screen">
|
||||||
@@ -87,27 +106,42 @@ export default function SeriesDetailPage({
|
|||||||
<h2 className="text-xl sm:text-2xl font-bold text-gray-900 dark:text-gray-100 truncate">
|
<h2 className="text-xl sm:text-2xl font-bold text-gray-900 dark:text-gray-100 truncate">
|
||||||
Books in Series
|
Books in Series
|
||||||
</h2>
|
</h2>
|
||||||
{series.books.length > 0 && (
|
{booksCountText && (
|
||||||
<span className="text-sm text-gray-600 dark:text-gray-400 hidden sm:inline whitespace-nowrap">
|
<span className="text-sm text-gray-600 dark:text-gray-400 hidden sm:inline whitespace-nowrap">
|
||||||
({series.books.length} title{series.books.length !== 1 ? 's' : ''})
|
({booksCountText})
|
||||||
</span>
|
</span>
|
||||||
)}
|
)}
|
||||||
<div className="ml-auto flex items-center gap-1">
|
<SectionToolbar
|
||||||
<SquareCoversToggle enabled={squareCovers} onToggle={setSquareCovers} />
|
hideAvailable={hideAvailable}
|
||||||
<CardSizeControls size={cardSize} onSizeChange={setCardSize} />
|
onToggleHideAvailable={setHideAvailable}
|
||||||
</div>
|
squareCovers={squareCovers}
|
||||||
|
onToggleSquareCovers={setSquareCovers}
|
||||||
|
cardSize={cardSize}
|
||||||
|
onCardSizeChange={setCardSize}
|
||||||
|
/>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
{/* Books Grid */}
|
{/* Books Grid */}
|
||||||
<AudiobookGrid
|
<AudiobookGrid
|
||||||
audiobooks={series.books}
|
audiobooks={filteredBooks}
|
||||||
isLoading={seriesLoading}
|
isLoading={seriesLoading}
|
||||||
emptyMessage={`No books found for ${series.title}`}
|
emptyMessage={`No books found for ${series.title}`}
|
||||||
cardSize={cardSize}
|
cardSize={cardSize}
|
||||||
squareCovers={squareCovers}
|
squareCovers={squareCovers}
|
||||||
/>
|
/>
|
||||||
|
|
||||||
|
{/* Load More Bar */}
|
||||||
|
{filteredBooks.length > 0 && (
|
||||||
|
<LoadMoreBar
|
||||||
|
loadedCount={filteredBooks.length}
|
||||||
|
totalCount={series.bookCount > 0 ? series.bookCount : undefined}
|
||||||
|
hasMore={hasMore}
|
||||||
|
isLoading={isLoadingMore}
|
||||||
|
onLoadMore={loadMore}
|
||||||
|
/>
|
||||||
|
)}
|
||||||
</div>
|
</div>
|
||||||
)}
|
)}
|
||||||
</main>
|
</main>
|
||||||
|
|||||||
@@ -134,6 +134,7 @@ export function BookDateStep({
|
|||||||
>
|
>
|
||||||
<option value="openai">OpenAI</option>
|
<option value="openai">OpenAI</option>
|
||||||
<option value="claude">Claude (Anthropic)</option>
|
<option value="claude">Claude (Anthropic)</option>
|
||||||
|
<option value="gemini">Google Gemini</option>
|
||||||
</select>
|
</select>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
@@ -152,7 +153,7 @@ export function BookDateStep({
|
|||||||
onUpdate('bookdateConfigured', false);
|
onUpdate('bookdateConfigured', false);
|
||||||
onUpdate('bookdateModels', []);
|
onUpdate('bookdateModels', []);
|
||||||
}}
|
}}
|
||||||
placeholder={bookdateProvider === 'openai' ? 'sk-...' : 'sk-ant-...'}
|
placeholder={bookdateProvider === 'openai' ? 'sk-...' : bookdateProvider === 'gemini' ? 'AIza...' : 'sk-ant-...'}
|
||||||
className="w-full px-4 py-2 border border-gray-300 dark:border-gray-600 rounded-lg bg-white dark:bg-gray-800 text-gray-900 dark:text-white focus:ring-2 focus:ring-blue-500"
|
className="w-full px-4 py-2 border border-gray-300 dark:border-gray-600 rounded-lg bg-white dark:bg-gray-800 text-gray-900 dark:text-white focus:ring-2 focus:ring-blue-500"
|
||||||
/>
|
/>
|
||||||
<p className="text-xs text-gray-500 dark:text-gray-400 mt-1">
|
<p className="text-xs text-gray-500 dark:text-gray-400 mt-1">
|
||||||
|
|||||||
Some files were not shown because too many files have changed in this diff Show More
Reference in New Issue
Block a user