Strip every comment from the source (parsed with the TypeScript compiler so strings, template literals, regex literals and JSX are never touched), and drop internal/working artifacts that do not belong in the public repository (design mockups, internal analysis docs, a stray backup file and an old log). No functional change: build is green, the full test suite passes.
196 lines
6.8 KiB
TypeScript
196 lines
6.8 KiB
TypeScript
import fs from "node:fs";
|
|
import path from "node:path";
|
|
import { AppSettings } from "../shared/types";
|
|
import { parseDebridLinkApiKeys } from "../shared/debrid-link-keys";
|
|
import { parseMegaDebridAccounts } from "../shared/mega-debrid-accounts";
|
|
import { StoragePaths } from "./storage";
|
|
|
|
export type HealthCheckSeverity = "INFO" | "WARN" | "ERROR";
|
|
|
|
export interface HealthCheckFinding {
|
|
severity: HealthCheckSeverity;
|
|
code: string;
|
|
message: string;
|
|
hint?: string;
|
|
}
|
|
|
|
export interface HealthCheckReport {
|
|
findings: HealthCheckFinding[];
|
|
errorCount: number;
|
|
warnCount: number;
|
|
infoCount: number;
|
|
}
|
|
|
|
const LOW_DISK_SPACE_BYTES = 5 * 1024 * 1024 * 1024;
|
|
const LARGE_STATE_FILE_BYTES = 50 * 1024 * 1024;
|
|
|
|
function safeExists(p: string): boolean {
|
|
try {
|
|
return fs.existsSync(p);
|
|
} catch {
|
|
return false;
|
|
}
|
|
}
|
|
|
|
function getFileSizeBytes(p: string): number {
|
|
try {
|
|
const stat = fs.statSync(p);
|
|
return stat.size;
|
|
} catch {
|
|
return 0;
|
|
}
|
|
}
|
|
|
|
function isWritable(dir: string): boolean {
|
|
const probe = path.join(dir, `.rddl-health-probe-${Date.now()}-${Math.random().toString(36).slice(2, 8)}`);
|
|
try {
|
|
fs.writeFileSync(probe, "x", { encoding: "utf8" });
|
|
fs.rmSync(probe, { force: true });
|
|
return true;
|
|
} catch {
|
|
return false;
|
|
}
|
|
}
|
|
|
|
function getFreeDiskSpaceBytes(target: string): number | null {
|
|
try {
|
|
const statfs = (fs as unknown as { statfsSync?: (p: string) => { bavail: bigint; bsize: bigint } }).statfsSync;
|
|
if (typeof statfs !== "function") {
|
|
return null;
|
|
}
|
|
const result = statfs(target);
|
|
const bavail = BigInt(result.bavail);
|
|
const bsize = BigInt(result.bsize);
|
|
const free = bavail * bsize;
|
|
if (free > BigInt(Number.MAX_SAFE_INTEGER)) {
|
|
return Number.MAX_SAFE_INTEGER;
|
|
}
|
|
return Number(free);
|
|
} catch {
|
|
return null;
|
|
}
|
|
}
|
|
|
|
function countConfiguredProviders(settings: AppSettings): { count: number; providers: string[] } {
|
|
const providers: string[] = [];
|
|
if (settings.token?.trim() || settings.realDebridUseWebLogin) {
|
|
providers.push("Real-Debrid");
|
|
}
|
|
if (settings.allDebridToken?.trim() || settings.allDebridUseWebLogin) {
|
|
providers.push("AllDebrid");
|
|
}
|
|
if (settings.bestToken?.trim() || settings.bestDebridUseWebLogin) {
|
|
providers.push("BestDebrid");
|
|
}
|
|
if (settings.oneFichierApiKey?.trim()) {
|
|
providers.push("1Fichier");
|
|
}
|
|
if (settings.ddownloadLogin?.trim() && settings.ddownloadPassword?.trim()) {
|
|
providers.push("DDownload");
|
|
}
|
|
if (settings.linkSnappyLogin?.trim() && settings.linkSnappyPassword?.trim()) {
|
|
providers.push("LinkSnappy");
|
|
}
|
|
const dlKeys = parseDebridLinkApiKeys(settings.debridLinkApiKeys || "");
|
|
if (dlKeys.length > 0) {
|
|
providers.push(`Debrid-Link (${dlKeys.length} Key${dlKeys.length === 1 ? "" : "s"})`);
|
|
}
|
|
const megaAccounts = parseMegaDebridAccounts(settings.megaCredentials || "");
|
|
const legacyMegaConfigured = Boolean(settings.megaLogin?.trim() && settings.megaPassword?.trim());
|
|
if (megaAccounts.length > 0) {
|
|
providers.push(`Mega-Debrid (${megaAccounts.length} Acc)`);
|
|
} else if (legacyMegaConfigured) {
|
|
providers.push("Mega-Debrid");
|
|
}
|
|
return { count: providers.length, providers };
|
|
}
|
|
|
|
export function runStartupHealthCheck(settings: AppSettings, storagePaths: StoragePaths): HealthCheckReport {
|
|
const findings: HealthCheckFinding[] = [];
|
|
|
|
const outputDir = String(settings.outputDir || "").trim();
|
|
if (!outputDir) {
|
|
findings.push({
|
|
severity: "WARN",
|
|
code: "outputDir_missing",
|
|
message: "Kein Download-Ziel-Verzeichnis konfiguriert",
|
|
hint: "In den Einstellungen unter 'Downloads' einen Ziel-Ordner setzen, sonst koennen keine Downloads starten."
|
|
});
|
|
} else if (!safeExists(outputDir)) {
|
|
findings.push({
|
|
severity: "WARN",
|
|
code: "outputDir_not_found",
|
|
message: `Download-Ziel-Ordner existiert nicht: ${outputDir}`,
|
|
hint: "Der Ordner wird beim ersten Download automatisch erstellt, sofern der Elternordner existiert und beschreibbar ist."
|
|
});
|
|
} else if (!isWritable(outputDir)) {
|
|
findings.push({
|
|
severity: "ERROR",
|
|
code: "outputDir_not_writable",
|
|
message: `Download-Ziel-Ordner ist NICHT beschreibbar: ${outputDir}`,
|
|
hint: "Rechte pruefen oder anderen Ordner waehlen. Downloads werden sonst direkt scheitern."
|
|
});
|
|
} else {
|
|
const freeBytes = getFreeDiskSpaceBytes(outputDir);
|
|
if (freeBytes !== null && freeBytes < LOW_DISK_SPACE_BYTES) {
|
|
const freeMb = Math.round(freeBytes / (1024 * 1024));
|
|
findings.push({
|
|
severity: "WARN",
|
|
code: "low_disk_space",
|
|
message: `Wenig freier Speicher im Download-Ordner: ~${freeMb} MB verfuegbar (Schwelle ${LOW_DISK_SPACE_BYTES / (1024 * 1024 * 1024)} GB)`,
|
|
hint: "Groessere Downloads koennen auf halbem Weg fehlschlagen. Vorher Platz schaffen oder anderen Ordner waehlen."
|
|
});
|
|
}
|
|
}
|
|
|
|
const { count, providers } = countConfiguredProviders(settings);
|
|
if (count === 0) {
|
|
findings.push({
|
|
severity: "WARN",
|
|
code: "no_provider_configured",
|
|
message: "Kein Debrid-Provider konfiguriert — Downloads werden nicht funktionieren",
|
|
hint: "In den Einstellungen mindestens einen Provider (Real-Debrid, Mega-Debrid, Debrid-Link, ...) einrichten."
|
|
});
|
|
} else {
|
|
findings.push({
|
|
severity: "INFO",
|
|
code: "providers_configured",
|
|
message: `Konfigurierte Provider: ${providers.join(", ")}`
|
|
});
|
|
}
|
|
|
|
if (safeExists(storagePaths.sessionFile)) {
|
|
const sizeBytes = getFileSizeBytes(storagePaths.sessionFile);
|
|
if (sizeBytes > LARGE_STATE_FILE_BYTES) {
|
|
const sizeMb = Math.round(sizeBytes / (1024 * 1024));
|
|
findings.push({
|
|
severity: "WARN",
|
|
code: "large_state_file",
|
|
message: `State-Datei ist sehr gross: ${sizeMb} MB (${path.basename(storagePaths.sessionFile)})`,
|
|
hint: "Alte abgeschlossene Pakete aus der Queue entfernen, damit Startup + Save schneller werden."
|
|
});
|
|
}
|
|
}
|
|
|
|
if (!safeExists(storagePaths.baseDir)) {
|
|
findings.push({
|
|
severity: "ERROR",
|
|
code: "baseDir_missing",
|
|
message: `Runtime-Verzeichnis existiert nicht: ${storagePaths.baseDir}`,
|
|
hint: "Ohne Runtime-Verzeichnis koennen weder Settings noch Session-State persistiert werden."
|
|
});
|
|
} else if (!isWritable(storagePaths.baseDir)) {
|
|
findings.push({
|
|
severity: "ERROR",
|
|
code: "baseDir_not_writable",
|
|
message: `Runtime-Verzeichnis ist NICHT beschreibbar: ${storagePaths.baseDir}`,
|
|
hint: "Rechte auf das Runtime-Verzeichnis pruefen (%APPDATA%/Real-Debrid-Downloader/runtime)."
|
|
});
|
|
}
|
|
|
|
const errorCount = findings.filter((f) => f.severity === "ERROR").length;
|
|
const warnCount = findings.filter((f) => f.severity === "WARN").length;
|
|
const infoCount = findings.filter((f) => f.severity === "INFO").length;
|
|
return { findings, errorCount, warnCount, infoCount };
|
|
}
|