Files
netfelix-audio-fix/server/types.ts
T
felixfoertsch 8112bfeb65
Build and Push Docker Image / build (push) Successful in 3m3s
per-track language override on audio detail page
adds stream_decisions.custom_language (ISO 639-2 code or null) so the
user can correct a mislabeled audio track — e.g. a Spanish dub tagged
"und" in the container — without going through Jellyfin. the override
wins over stream.language everywhere it matters: the analyzer reads it
for keep/remove decisions and track ordering, the ffmpeg command builder
writes it as both the language metadata tag and the harmonized track
title, and reanalyze preserves it across reruns and rescans.

on the audio detail page, each pending audio row swaps its language
cell for an inline <select> populated from LANG_NAMES. picking the raw
file language clears the override; anything else sets it and triggers a
server-side reanalyze so keep/remove + target_index update immediately.
a small ✎ hint marks overridden tracks. rebuilt commands tag the output
accordingly so Jellyfin reads the corrected language.

PATCH /api/review/:id/stream/:streamId/language validates the code
against LANG_NAMES (accepts ISO 639-1/2/2B aliases, rejects garbage)
and runs reanalyze inside.

Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
2026-04-20 00:05:31 +02:00

186 lines
5.3 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;
auto_class: "auto" | "auto_heuristic" | "manual" | null;
sorted: number;
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 StreamDecision {
id: number;
plan_id: number;
stream_id: number;
action: "keep" | "remove";
target_index: number | null;
custom_title: string | null;
/** Per-stream language override. When set, the analyzer and ffmpeg
* command builder both read this in preference to the raw
* media_streams.language. Lets the user correct an "und" or
* mislabeled audio track without going through Jellyfin. */
custom_language: 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;
auto_class: "auto" | "auto_heuristic" | "manual";
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>;
SeriesProviderIds?: 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;
}