Add GET /stats API endpoint with collection overview, reading status, format/library breakdowns, top series, and monthly additions. Replace static home page with interactive dashboard featuring donut charts, bar charts, and progress bars. Use distinct colors for series (warning/yellow) across nav, page titles, and quick links. Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>
171 lines
6.3 KiB
Rust
171 lines
6.3 KiB
Rust
use utoipa::openapi::security::{HttpAuthScheme, HttpBuilder, SecurityScheme};
|
|
use utoipa::OpenApi;
|
|
|
|
#[derive(OpenApi)]
|
|
#[openapi(
|
|
paths(
|
|
crate::books::list_books,
|
|
crate::books::get_book,
|
|
crate::reading_progress::get_reading_progress,
|
|
crate::reading_progress::update_reading_progress,
|
|
crate::reading_progress::mark_series_read,
|
|
crate::books::get_thumbnail,
|
|
crate::books::list_series,
|
|
crate::books::list_all_series,
|
|
crate::books::ongoing_series,
|
|
crate::books::ongoing_books,
|
|
crate::books::convert_book,
|
|
crate::pages::get_page,
|
|
crate::search::search_books,
|
|
crate::index_jobs::enqueue_rebuild,
|
|
crate::thumbnails::start_thumbnails_rebuild,
|
|
crate::thumbnails::start_thumbnails_regenerate,
|
|
crate::index_jobs::list_index_jobs,
|
|
crate::index_jobs::get_active_jobs,
|
|
crate::index_jobs::get_job_details,
|
|
crate::index_jobs::stream_job_progress,
|
|
crate::index_jobs::get_job_errors,
|
|
crate::index_jobs::cancel_job,
|
|
crate::index_jobs::list_folders,
|
|
crate::libraries::list_libraries,
|
|
crate::libraries::create_library,
|
|
crate::libraries::delete_library,
|
|
crate::libraries::scan_library,
|
|
crate::libraries::update_monitoring,
|
|
crate::tokens::list_tokens,
|
|
crate::tokens::create_token,
|
|
crate::tokens::revoke_token,
|
|
crate::tokens::delete_token,
|
|
crate::stats::get_stats,
|
|
crate::settings::get_settings,
|
|
crate::settings::get_setting,
|
|
crate::settings::update_setting,
|
|
crate::settings::clear_cache,
|
|
crate::settings::get_cache_stats,
|
|
crate::settings::get_thumbnail_stats,
|
|
),
|
|
components(
|
|
schemas(
|
|
crate::books::ListBooksQuery,
|
|
crate::books::BookItem,
|
|
crate::books::BooksPage,
|
|
crate::books::BookDetails,
|
|
crate::reading_progress::ReadingProgressResponse,
|
|
crate::reading_progress::UpdateReadingProgressRequest,
|
|
crate::reading_progress::MarkSeriesReadRequest,
|
|
crate::reading_progress::MarkSeriesReadResponse,
|
|
crate::books::SeriesItem,
|
|
crate::books::SeriesPage,
|
|
crate::books::ListAllSeriesQuery,
|
|
crate::books::OngoingQuery,
|
|
crate::pages::PageQuery,
|
|
crate::search::SearchQuery,
|
|
crate::search::SearchResponse,
|
|
crate::search::SeriesHit,
|
|
crate::index_jobs::RebuildRequest,
|
|
crate::thumbnails::ThumbnailsRebuildRequest,
|
|
crate::index_jobs::IndexJobResponse,
|
|
crate::index_jobs::IndexJobDetailResponse,
|
|
crate::index_jobs::JobErrorResponse,
|
|
crate::index_jobs::ProgressEvent,
|
|
crate::index_jobs::FolderItem,
|
|
crate::libraries::LibraryResponse,
|
|
crate::libraries::CreateLibraryRequest,
|
|
crate::libraries::UpdateMonitoringRequest,
|
|
crate::tokens::CreateTokenRequest,
|
|
crate::tokens::TokenResponse,
|
|
crate::tokens::CreatedTokenResponse,
|
|
crate::settings::UpdateSettingRequest,
|
|
crate::settings::ClearCacheResponse,
|
|
crate::settings::CacheStats,
|
|
crate::settings::ThumbnailStats,
|
|
crate::stats::StatsResponse,
|
|
crate::stats::StatsOverview,
|
|
crate::stats::ReadingStatusStats,
|
|
crate::stats::FormatCount,
|
|
crate::stats::LanguageCount,
|
|
crate::stats::LibraryStats,
|
|
crate::stats::TopSeries,
|
|
crate::stats::MonthlyAdditions,
|
|
ErrorResponse,
|
|
)
|
|
),
|
|
security(
|
|
("Bearer" = [])
|
|
),
|
|
tags(
|
|
(name = "books", description = "Read-only endpoints for browsing and searching books"),
|
|
(name = "reading-progress", description = "Reading progress tracking per book"),
|
|
(name = "libraries", description = "Library management endpoints (Admin only)"),
|
|
(name = "indexing", description = "Search index management and job control (Admin only)"),
|
|
(name = "tokens", description = "API token management (Admin only)"),
|
|
(name = "settings", description = "Application settings and cache management (Admin only)"),
|
|
),
|
|
modifiers(&SecurityAddon)
|
|
)]
|
|
pub struct ApiDoc;
|
|
|
|
pub struct SecurityAddon;
|
|
|
|
impl utoipa::Modify for SecurityAddon {
|
|
fn modify(&self, openapi: &mut utoipa::openapi::OpenApi) {
|
|
if let Some(components) = openapi.components.as_mut() {
|
|
components.add_security_scheme(
|
|
"Bearer",
|
|
SecurityScheme::Http(
|
|
HttpBuilder::new()
|
|
.scheme(HttpAuthScheme::Bearer)
|
|
.bearer_format("JWT")
|
|
.description(Some(
|
|
"Enter your API Bearer token (format: stl_<prefix>_<secret>)",
|
|
))
|
|
.build(),
|
|
),
|
|
);
|
|
}
|
|
}
|
|
}
|
|
|
|
#[derive(utoipa::ToSchema)]
|
|
pub struct ErrorResponse {
|
|
#[allow(dead_code)]
|
|
pub error: String,
|
|
}
|
|
|
|
#[cfg(test)]
|
|
mod tests {
|
|
use super::*;
|
|
use utoipa::OpenApi;
|
|
|
|
#[test]
|
|
fn test_openapi_generation() {
|
|
let api_doc = ApiDoc::openapi();
|
|
let json = api_doc
|
|
.to_pretty_json()
|
|
.expect("Failed to serialize OpenAPI");
|
|
|
|
// Check that all $ref targets exist in components/schemas
|
|
let doc: serde_json::Value =
|
|
serde_json::from_str(&json).expect("OpenAPI JSON should be valid");
|
|
let empty = serde_json::Map::new();
|
|
let schemas = doc["components"]["schemas"]
|
|
.as_object()
|
|
.unwrap_or(&empty);
|
|
let prefix = "#/components/schemas/";
|
|
let mut broken: Vec<String> = Vec::new();
|
|
for part in json.split(prefix).skip(1) {
|
|
if let Some(name) = part.split('"').next() {
|
|
if !schemas.contains_key(name) {
|
|
broken.push(name.to_string());
|
|
}
|
|
}
|
|
}
|
|
broken.dedup();
|
|
assert!(broken.is_empty(), "Unresolved schema refs: {:?}", broken);
|
|
|
|
// Save to file for inspection
|
|
std::fs::write("/tmp/openapi.json", &json).expect("Failed to write file");
|
|
println!("OpenAPI JSON saved to /tmp/openapi.json");
|
|
}
|
|
}
|