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:
parent
34a1a59a2a
commit
664e34fc53
@ -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(),
|
||||||
|
|||||||
Loading…
Reference in New Issue
Block a user