Test: dedizierter Per-Account-Timeout-Rotationstest (acc1 haengt -> acc2 versucht)

Schliesst die Coverage-Luecke zum v1.7.168-Fix (c4a49d9): der korrigierte Abort-Test
prueft nur Signal-Propagation, nicht die Kernlogik des Fix. Dieser Test (Fake-Timer)
faehrt den echten Rotationspfad: acc1 haengt bis sein Per-Account-Timeout feuert ->
30s-Cooldown -> Loop probiert acc2 -> Erfolg. Ohne den Fix (keine Abort-Quelle bei
fehlendem globalem Signal) wuerde acc1 ewig haengen -> Test-Timeout. 642 Tests gruen.

Co-Authored-By: Claude Opus 4.8 (1M context) <noreply@anthropic.com>
This commit is contained in:
Sucukdeluxe 2026-05-30 23:44:19 +02:00
parent 34a1a59a2a
commit 664e34fc53

View File

@ -1400,6 +1400,62 @@ describe("debrid service", () => {
} }
}); });
it("rotation per-account timeout: a hanging account is skipped so the next account is tried", async () => {
// Kern des v1.7.168-Fix: haengt Account 1, darf das NICHT die ganze Rotation
// fressen — der Per-Account-Timeout (PER_ACCOUNT_ATTEMPT_TIMEOUT_MS, default 25s)
// bricht NUR acc1 ab (kurzer Cooldown), der Loop probiert acc2. Ohne globales
// Signal (kein download-manager-Wrap) testet das den Per-Account-Timeout isoliert.
const settings = {
...defaultSettings(),
token: "",
bestToken: "",
allDebridToken: "",
megaLogin: "user1",
megaPassword: "pass1",
megaCredentials: "user1:pass1\nuser2:pass2",
providerOrder: [] as const,
providerPrimary: "megadebrid" as const,
providerSecondary: "none" as const,
providerTertiary: "none" as const,
autoProviderFallback: false
};
// API connect schlaegt pro Account schnell fehl (500) -> Web-Fallback (megaWeb).
globalThis.fetch = (async () => new Response("error", { status: 500 })) as typeof fetch;
let call = 0;
const megaWeb = vi.fn((_link: string, signal?: AbortSignal) => {
call += 1;
if (call === 1) {
// Account 1 haengt, bis sein eigener Per-Account-Timeout das Signal abortet.
return new Promise<never>((_, reject) => {
if (signal?.aborted) { reject(new Error("aborted:acc1-hang")); return; }
signal?.addEventListener("abort", () => reject(new Error("aborted:acc1-hang")), { once: true });
});
}
// Account 2 liefert sofort.
return Promise.resolve({
fileName: "acc2.rar",
directUrl: "https://mega-web.example/acc2.rar",
fileSize: null,
retriesUsed: 0
});
});
const service = new DebridService(settings, { megaWebUnrestrict: megaWeb });
vi.useFakeTimers();
try {
const pending = service.unrestrictLink("https://rapidgator.net/file/rotate-acc");
// Per-Account-Timeout von acc1 (25s) feuern lassen -> acc1 faellt -> Loop -> acc2.
await vi.advanceTimersByTimeAsync(30000);
const result = await pending;
expect(result.directUrl).toBe("https://mega-web.example/acc2.rar");
expect(megaWeb).toHaveBeenCalledTimes(2);
} finally {
vi.useRealTimers();
}
});
it("respects provider selection and does not append hidden providers", async () => { it("respects provider selection and does not append hidden providers", async () => {
const settings = { const settings = {
...defaultSettings(), ...defaultSettings(),