Some checks failed
Build and Push Docker Image / build (push) Failing after 16s
ffmpeg now writes -metadata:s:a:i language=<iso3> on every kept audio track so
files end up with canonical 3-letter tags (en → eng, ger → deu, null → und).
analyzer passes stream.profile (not title) to transcodeTarget so lossless
dts-hd ma in mkv correctly targets flac. is_noop also checks og-is-default and
canonical-language so pipeline-would-change-it cases stop showing as done.
normalizeLanguage gains 2→3 mapping, and mapStream no longer normalizes at
ingest so the raw jellyfin tag survives for the canonical check.
per-item scan work runs in a single db.transaction for large sqlite speedups,
extracted into server/services/rescan.ts so execute.ts can reuse it.
on successful job, execute calls jellyfin /Items/{id}/Refresh, waits for
DateLastRefreshed to change, refetches the item, and upserts it through the
same pipeline; plan flips to done iff the fresh streams satisfy is_noop.
schema wiped + rewritten to carry jellyfin_raw, external_raw, profile,
bit_depth, date_last_refreshed, runtime_ticks, original_title, last_executed_at
— so future scans aren't required to stay correct. user must drop data/*.db.
Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
191 lines
5.2 KiB
TypeScript
191 lines
5.2 KiB
TypeScript
// ─── Database row types ───────────────────────────────────────────────────────
|
|
|
|
export interface MediaItem {
|
|
id: number;
|
|
jellyfin_id: string;
|
|
type: "Movie" | "Episode";
|
|
name: string;
|
|
original_title: string | null;
|
|
series_name: string | null;
|
|
series_jellyfin_id: string | null;
|
|
season_number: number | null;
|
|
episode_number: number | null;
|
|
year: number | null;
|
|
file_path: string;
|
|
file_size: number | null;
|
|
container: string | null;
|
|
runtime_ticks: number | null;
|
|
date_last_refreshed: string | null;
|
|
original_language: string | null;
|
|
orig_lang_source: "jellyfin" | "radarr" | "sonarr" | "manual" | null;
|
|
needs_review: number;
|
|
imdb_id: string | null;
|
|
tmdb_id: string | null;
|
|
tvdb_id: string | null;
|
|
jellyfin_raw: string | null;
|
|
external_raw: string | null;
|
|
scan_status: "pending" | "scanned" | "error";
|
|
scan_error: string | null;
|
|
last_scanned_at: string | null;
|
|
last_executed_at: string | null;
|
|
created_at: string;
|
|
}
|
|
|
|
export interface MediaStream {
|
|
id: number;
|
|
item_id: number;
|
|
stream_index: number;
|
|
type: "Video" | "Audio" | "Subtitle" | "Data" | "EmbeddedImage";
|
|
codec: string | null;
|
|
profile: string | null;
|
|
/** Raw language tag as reported by Jellyfin (e.g. "en", "eng", "ger", null).
|
|
* Not normalized on ingest — callers use normalizeLanguage() for comparison
|
|
* so we can detect non-canonical tags that the pipeline should rewrite. */
|
|
language: string | null;
|
|
language_display: string | null;
|
|
title: string | null;
|
|
is_default: number;
|
|
is_forced: number;
|
|
is_hearing_impaired: number;
|
|
channels: number | null;
|
|
channel_layout: string | null;
|
|
bit_rate: number | null;
|
|
sample_rate: number | null;
|
|
bit_depth: number | null;
|
|
}
|
|
|
|
export interface ReviewPlan {
|
|
id: number;
|
|
item_id: number;
|
|
status: "pending" | "approved" | "skipped" | "done" | "error";
|
|
is_noop: number;
|
|
confidence: "high" | "low";
|
|
apple_compat: "direct_play" | "remux" | "audio_transcode" | null;
|
|
job_type: "copy" | "transcode";
|
|
subs_extracted: number;
|
|
notes: string | null;
|
|
reviewed_at: string | null;
|
|
created_at: string;
|
|
}
|
|
|
|
export interface SubtitleFile {
|
|
id: number;
|
|
item_id: number;
|
|
file_path: string;
|
|
language: string | null;
|
|
codec: string | null;
|
|
is_forced: number;
|
|
is_hearing_impaired: number;
|
|
file_size: number | null;
|
|
created_at: string;
|
|
}
|
|
|
|
export interface StreamDecision {
|
|
id: number;
|
|
plan_id: number;
|
|
stream_id: number;
|
|
action: "keep" | "remove";
|
|
target_index: number | null;
|
|
custom_title: string | null;
|
|
transcode_codec: string | null;
|
|
}
|
|
|
|
export interface Job {
|
|
id: number;
|
|
item_id: number;
|
|
command: string;
|
|
job_type: "copy" | "transcode";
|
|
status: "pending" | "running" | "done" | "error";
|
|
output: string | null;
|
|
exit_code: number | null;
|
|
created_at: string;
|
|
started_at: string | null;
|
|
completed_at: string | null;
|
|
}
|
|
|
|
// ─── Analyzer types ───────────────────────────────────────────────────────────
|
|
|
|
export interface StreamWithDecision extends MediaStream {
|
|
action: "keep" | "remove";
|
|
target_index: number | null;
|
|
}
|
|
|
|
export interface PlanResult {
|
|
is_noop: boolean;
|
|
has_subs: boolean;
|
|
confidence: "high" | "low";
|
|
apple_compat: "direct_play" | "remux" | "audio_transcode" | null;
|
|
job_type: "copy" | "transcode";
|
|
decisions: Array<{
|
|
stream_id: number;
|
|
action: "keep" | "remove";
|
|
target_index: number | null;
|
|
transcode_codec: string | null;
|
|
}>;
|
|
notes: string[];
|
|
}
|
|
|
|
// ─── Jellyfin API types ───────────────────────────────────────────────────────
|
|
|
|
export interface JellyfinMediaStream {
|
|
Type: string;
|
|
Index: number;
|
|
Codec?: string;
|
|
Profile?: string;
|
|
Language?: string;
|
|
DisplayLanguage?: string;
|
|
Title?: string;
|
|
IsDefault?: boolean;
|
|
IsForced?: boolean;
|
|
IsHearingImpaired?: boolean;
|
|
IsExternal?: boolean;
|
|
Channels?: number;
|
|
ChannelLayout?: string;
|
|
BitRate?: number;
|
|
SampleRate?: number;
|
|
BitDepth?: number;
|
|
}
|
|
|
|
export interface JellyfinItem {
|
|
Id: string;
|
|
Type: string;
|
|
Name: string;
|
|
OriginalTitle?: string;
|
|
SeriesName?: string;
|
|
SeriesId?: string;
|
|
ParentIndexNumber?: number;
|
|
IndexNumber?: number;
|
|
ProductionYear?: number;
|
|
Path?: string;
|
|
Size?: number;
|
|
Container?: string;
|
|
RunTimeTicks?: number;
|
|
DateLastRefreshed?: string;
|
|
MediaStreams?: JellyfinMediaStream[];
|
|
ProviderIds?: Record<string, string>;
|
|
}
|
|
|
|
export interface JellyfinUser {
|
|
Id: string;
|
|
Name: string;
|
|
}
|
|
|
|
// ─── Scan state ───────────────────────────────────────────────────────────────
|
|
|
|
export interface ScanProgress {
|
|
total: number;
|
|
scanned: number;
|
|
current_item: string;
|
|
errors: number;
|
|
running: boolean;
|
|
}
|
|
|
|
// ─── SSE event helpers ────────────────────────────────────────────────────────
|
|
|
|
export type SseEventType = "progress" | "log" | "complete" | "error";
|
|
|
|
export interface SseEvent {
|
|
type: SseEventType;
|
|
data: unknown;
|
|
}
|