From 577be4ea3440c258689a560d3d1a6c2ae158592b Mon Sep 17 00:00:00 2001 From: Fuwn Date: Fri, 5 Jun 2026 14:48:42 +0000 Subject: fix(airing): roll just-aired releases forward to stop refresh loop nextReleaseTime kept a just-aired release in the past for 5 minutes, showing a negative countdown and driving scheduleAiringRefresh into a 1-second revalidate loop. Always roll a past release to its next occurrence, and only arm the refresh timer on future airings. --- src/lib/Media/Anime/Airing/classify.test.ts | 24 ++++++++++++++++++++++++ 1 file changed, 24 insertions(+) (limited to 'src/lib/Media/Anime/Airing/classify.test.ts') diff --git a/src/lib/Media/Anime/Airing/classify.test.ts b/src/lib/Media/Anime/Airing/classify.test.ts index eeff2036..7b3aea32 100644 --- a/src/lib/Media/Anime/Airing/classify.test.ts +++ b/src/lib/Media/Anime/Airing/classify.test.ts @@ -210,6 +210,30 @@ describe("countdown source fallback", () => { }); }); +describe("release timing", () => { + it("rolls a just-aired release forward instead of returning a past time", () => { + clearInjectAiringTimeCache(); + + const media = baseMedia(170019); + const justAired = Math.floor(Date.now() / 1000) - 3 * 60; + + media.nextAiringEpisode = { + episode: 10, + airingAt: justAired + 12 * 60 * 60, + }; + + const schedule = subScheduleFor(media, justAired); + + settings.setKey("countdownSource", "sub"); + + const injected = injectAiringTime(media, schedule); + + expect(injected.nextAiringEpisode?.airingAt).toBeGreaterThan( + Date.now() / 1000, + ); + }); +}); + describe("injectAiringTime cache safety", () => { it("does not let caller mutation poison cached injected media", () => { const media = baseMedia(444444); -- cgit v1.2.3