Some checks failed
CI / Backend (push) Failing after 11s
Release / Check ui (push) Failing after 51s
Release / Docker / ui (push) Has been skipped
CI / UI (push) Failing after 55s
Release / Test backend (push) Failing after 1m9s
Release / Docker / backend (push) Has been skipped
Release / Docker / runner (push) Has been skipped
Release / Docker / caddy (push) Failing after 28s
Release / Gitea Release (push) Has been skipped
CI / UI (pull_request) Failing after 42s
CI / Backend (pull_request) Successful in 3m45s
- LibreTranslate client (chunks on blank lines, ≤4500 chars, 3-goroutine semaphore)
- Runner translation task loop (OTel, heartbeat, MinIO storage)
- PocketBase translation_jobs collection support (create/claim/finish/list)
- Per-chapter language switcher on chapter reader (EN/RU/ID/PT/FR, polls until done)
- Admin /admin/translation page: bulk enqueue form + live-polling jobs table
- New backend routes: POST /api/translation/{slug}/{n}, GET /api/translation/status,
GET /api/translation/{slug}/{n}, GET /api/admin/translation/jobs,
POST /api/admin/translation/bulk
- ListTranslationTasks added to taskqueue.Reader interface + store impl
- All builds and tests pass; svelte-check: 0 errors
172 lines
6.9 KiB
Go
172 lines
6.9 KiB
Go
// Package domain contains the core value types shared across all packages
|
|
// in this module. It has zero internal imports — only the standard library.
|
|
// Every other package imports domain; domain imports nothing from this module.
|
|
package domain
|
|
|
|
import "time"
|
|
|
|
// ── Book types ────────────────────────────────────────────────────────────────
|
|
|
|
// BookMeta carries all bibliographic information about a novel.
|
|
type BookMeta struct {
|
|
Slug string `json:"slug"`
|
|
Title string `json:"title"`
|
|
Author string `json:"author"`
|
|
Cover string `json:"cover,omitempty"`
|
|
Status string `json:"status,omitempty"`
|
|
Genres []string `json:"genres,omitempty"`
|
|
Summary string `json:"summary,omitempty"`
|
|
TotalChapters int `json:"total_chapters,omitempty"`
|
|
SourceURL string `json:"source_url"`
|
|
Ranking int `json:"ranking,omitempty"`
|
|
Rating float64 `json:"rating,omitempty"`
|
|
// MetaUpdated is the Unix timestamp (seconds) when the book record was last
|
|
// updated in PocketBase. Populated on read; not sent on write (PocketBase
|
|
// manages its own updated field).
|
|
MetaUpdated int64 `json:"meta_updated,omitempty"`
|
|
}
|
|
|
|
// CatalogueEntry is a lightweight book reference returned by catalogue pages.
|
|
type CatalogueEntry struct {
|
|
Slug string `json:"slug"`
|
|
Title string `json:"title"`
|
|
URL string `json:"url"`
|
|
}
|
|
|
|
// ChapterRef is a reference to a single chapter returned by chapter-list pages.
|
|
type ChapterRef struct {
|
|
Number int `json:"number"`
|
|
Title string `json:"title"`
|
|
URL string `json:"url"`
|
|
Volume int `json:"volume,omitempty"`
|
|
}
|
|
|
|
// Chapter contains the fully-extracted text of a single chapter.
|
|
type Chapter struct {
|
|
Ref ChapterRef `json:"ref"`
|
|
Text string `json:"text"`
|
|
}
|
|
|
|
// RankingItem represents a single entry in the novel ranking list.
|
|
type RankingItem struct {
|
|
Rank int `json:"rank"`
|
|
Slug string `json:"slug"`
|
|
Title string `json:"title"`
|
|
Author string `json:"author,omitempty"`
|
|
Cover string `json:"cover,omitempty"`
|
|
Status string `json:"status,omitempty"`
|
|
Genres []string `json:"genres,omitempty"`
|
|
SourceURL string `json:"source_url,omitempty"`
|
|
Updated time.Time `json:"updated,omitempty"`
|
|
}
|
|
|
|
// ── Voice types ───────────────────────────────────────────────────────────────
|
|
|
|
// Voice describes a single text-to-speech voice available in the system.
|
|
type Voice struct {
|
|
// ID is the voice identifier passed to TTS clients (e.g. "af_bella", "alba").
|
|
ID string `json:"id"`
|
|
// Engine is "kokoro" or "pocket-tts".
|
|
Engine string `json:"engine"`
|
|
// Lang is the primary language tag (e.g. "en-us", "en-gb", "en", "es", "fr").
|
|
Lang string `json:"lang"`
|
|
// Gender is "f" or "m".
|
|
Gender string `json:"gender"`
|
|
}
|
|
|
|
// ── Storage record types ──────────────────────────────────────────────────────
|
|
|
|
// ChapterInfo is a lightweight chapter descriptor stored in the index.
|
|
type ChapterInfo struct {
|
|
Number int `json:"number"`
|
|
Title string `json:"title"`
|
|
Date string `json:"date,omitempty"`
|
|
}
|
|
|
|
// ReadingProgress holds a single user's reading position for one book.
|
|
type ReadingProgress struct {
|
|
Slug string `json:"slug"`
|
|
Chapter int `json:"chapter"`
|
|
UpdatedAt time.Time `json:"updated_at"`
|
|
}
|
|
|
|
// ── Task record types ─────────────────────────────────────────────────────────
|
|
|
|
// TaskStatus enumerates the lifecycle states of any task.
|
|
type TaskStatus string
|
|
|
|
const (
|
|
TaskStatusPending TaskStatus = "pending"
|
|
TaskStatusRunning TaskStatus = "running"
|
|
TaskStatusDone TaskStatus = "done"
|
|
TaskStatusFailed TaskStatus = "failed"
|
|
TaskStatusCancelled TaskStatus = "cancelled"
|
|
)
|
|
|
|
// ScrapeTask represents a book-scraping job stored in PocketBase.
|
|
type ScrapeTask struct {
|
|
ID string `json:"id"`
|
|
Kind string `json:"kind"` // "catalogue" | "book" | "book_range"
|
|
TargetURL string `json:"target_url"` // non-empty for single-book tasks
|
|
FromChapter int `json:"from_chapter,omitempty"`
|
|
ToChapter int `json:"to_chapter,omitempty"`
|
|
WorkerID string `json:"worker_id,omitempty"`
|
|
Status TaskStatus `json:"status"`
|
|
BooksFound int `json:"books_found"`
|
|
ChaptersScraped int `json:"chapters_scraped"`
|
|
ChaptersSkipped int `json:"chapters_skipped"`
|
|
Errors int `json:"errors"`
|
|
Started time.Time `json:"started"`
|
|
Finished time.Time `json:"finished,omitempty"`
|
|
ErrorMessage string `json:"error_message,omitempty"`
|
|
}
|
|
|
|
// ScrapeResult is the outcome reported by the runner after finishing a ScrapeTask.
|
|
type ScrapeResult struct {
|
|
BooksFound int `json:"books_found"`
|
|
ChaptersScraped int `json:"chapters_scraped"`
|
|
ChaptersSkipped int `json:"chapters_skipped"`
|
|
Errors int `json:"errors"`
|
|
ErrorMessage string `json:"error_message,omitempty"`
|
|
}
|
|
|
|
// AudioTask represents an audio-generation job stored in PocketBase.
|
|
type AudioTask struct {
|
|
ID string `json:"id"`
|
|
CacheKey string `json:"cache_key"` // "slug/chapter/voice"
|
|
Slug string `json:"slug"`
|
|
Chapter int `json:"chapter"`
|
|
Voice string `json:"voice"`
|
|
WorkerID string `json:"worker_id,omitempty"`
|
|
Status TaskStatus `json:"status"`
|
|
ErrorMessage string `json:"error_message,omitempty"`
|
|
Started time.Time `json:"started"`
|
|
Finished time.Time `json:"finished,omitempty"`
|
|
}
|
|
|
|
// AudioResult is the outcome reported by the runner after finishing an AudioTask.
|
|
type AudioResult struct {
|
|
ObjectKey string `json:"object_key,omitempty"`
|
|
ErrorMessage string `json:"error_message,omitempty"`
|
|
}
|
|
|
|
// TranslationTask represents a machine-translation job stored in PocketBase.
|
|
type TranslationTask struct {
|
|
ID string `json:"id"`
|
|
CacheKey string `json:"cache_key"` // "{slug}/{chapter}/{lang}"
|
|
Slug string `json:"slug"`
|
|
Chapter int `json:"chapter"`
|
|
Lang string `json:"lang"`
|
|
WorkerID string `json:"worker_id,omitempty"`
|
|
Status TaskStatus `json:"status"`
|
|
ErrorMessage string `json:"error_message,omitempty"`
|
|
Started time.Time `json:"started"`
|
|
Finished time.Time `json:"finished,omitempty"`
|
|
}
|
|
|
|
// TranslationResult is the outcome reported by the runner after finishing a TranslationTask.
|
|
type TranslationResult struct {
|
|
ObjectKey string `json:"object_key,omitempty"`
|
|
ErrorMessage string `json:"error_message,omitempty"`
|
|
}
|