Fix stale extract pending states

This commit is contained in:
Sucukdeluxe 2026-03-09 20:14:35 +01:00
parent ef26237b3e
commit e6b8ea0abe
2 changed files with 95 additions and 15 deletions

View File

@ -589,6 +589,13 @@ function isExtractErrorLabel(statusText: string): boolean {
|| /^entpacken\b.*\btimeout\b/i.test(text);
}
function isTransientExtractStatus(statusText: string): boolean {
const text = String(statusText || "").trim();
return /^entpacken\b/i.test(text)
|| /^passwort\b/i.test(text)
|| /^finalisieren\b/i.test(text);
}
function shouldAutoRetryExtraction(statusText: string): boolean {
return !isExtractedLabel(statusText) && !isExtractErrorLabel(statusText);
}
@ -1102,11 +1109,13 @@ export function buildAutoRenameBaseNameFromFoldersWithOptions(
}
export function resolveArchiveItemsFromList(archiveName: string, items: DownloadItem[]): DownloadItem[] {
const entryLower = archiveName.toLowerCase();
const normalizeArchiveMatchName = (value: string): string =>
path.basename(String(value || "")).replace(/ \(\d+\)(?=\.[^.]+$)/, "");
const entryLower = normalizeArchiveMatchName(archiveName).toLowerCase();
// Helper: get item basename (try targetPath first, then fileName)
const itemBaseName = (item: DownloadItem): string =>
path.basename(item.targetPath || item.fileName || "");
normalizeArchiveMatchName(item.targetPath || item.fileName || "");
// Try pattern-based matching first (for multipart archives)
let pattern: RegExp | null = null;
@ -5296,9 +5305,11 @@ export class DownloadManager extends EventEmitter {
continue;
}
if (/^Entpacken\b/i.test(currentStatus) || /^Passwort\b/i.test(currentStatus) || /^Finalisieren\b/i.test(currentStatus)) {
if (isTransientExtractStatus(currentStatus)) {
const previousStatus = String(previousStatuses.get(entry.id) || "").trim();
entry.fullStatus = previousStatus || `Fertig (${humanSize(entry.downloadedBytes)})`;
entry.fullStatus = isTransientExtractStatus(previousStatus)
? `Fertig (${humanSize(entry.downloadedBytes)})`
: previousStatus || `Fertig (${humanSize(entry.downloadedBytes)})`;
entry.updatedAt = appliedAt;
}
}
@ -9719,16 +9730,6 @@ export class DownloadManager extends EventEmitter {
this.emitState();
};
// Mark all items as pending before extraction starts
for (const entry of completedItems) {
if (!isExtractedLabel(entry.fullStatus)) {
preExtractStatuses.set(entry.id, String(entry.fullStatus || "").trim());
entry.fullStatus = "Entpacken - Ausstehend";
entry.updatedAt = nowMs();
}
}
this.emitState();
const extractTimeoutMs = getPostExtractTimeoutMs();
const extractAbortController = new AbortController();
let timedOut = false;
@ -9774,6 +9775,23 @@ export class DownloadManager extends EventEmitter {
}
const fullArchiveSet = await this.findFullExtractArchiveSet(pkg, completedItems);
const fullExtractItemIds = new Set<string>();
for (const archivePath of fullArchiveSet) {
const archiveItems = resolveArchiveItems(path.basename(archivePath));
for (const entry of archiveItems) {
fullExtractItemIds.add(entry.id);
}
}
const pendingAt = nowMs();
for (const entry of completedItems) {
if (!fullExtractItemIds.has(entry.id) || isExtractedLabel(entry.fullStatus)) {
continue;
}
preExtractStatuses.set(entry.id, String(entry.fullStatus || "").trim());
entry.fullStatus = "Entpacken - Ausstehend";
entry.updatedAt = pendingAt;
}
this.emitState();
const result = await extractPackageArchives({
packageDir: pkg.outputDir,
targetDir: pkg.extractDir,

View File

@ -5,7 +5,7 @@ import http from "node:http";
import { EventEmitter, once } from "node:events";
import AdmZip from "adm-zip";
import { afterEach, describe, expect, it, vi } from "vitest";
import { DownloadManager, extractArchiveNameFromExtractorLogMessage, getAuthoritativeRealDebridTotal } from "../src/main/download-manager";
import { DownloadManager, extractArchiveNameFromExtractorLogMessage, getAuthoritativeRealDebridTotal, resolveArchiveItemsFromList } from "../src/main/download-manager";
import { defaultSettings } from "../src/main/constants";
import { parseDebridLinkApiKeys } from "../src/shared/debrid-link-keys";
import { getProviderUsageDayKey } from "../src/shared/provider-daily-limits";
@ -29,6 +29,20 @@ describe("extractArchiveNameFromExtractorLogMessage", () => {
});
});
describe("resolveArchiveItemsFromList", () => {
it("includes duplicate-suffixed archive copies in multipart matches", () => {
const items = [
{ id: "dup-1", fileName: "show.s01e26.part1.rar" },
{ id: "dup-2", fileName: "show.s01e26.part1.rar", targetPath: "C:\\Downloads\\show.s01e26.part1 (1).rar" },
{ id: "dup-3", fileName: "show.s01e26.part2.rar" }
] as any[];
const resolved = resolveArchiveItemsFromList("show.s01e26.part1.rar", items);
expect(resolved.map((item) => item.id)).toEqual(["dup-1", "dup-2", "dup-3"]);
});
});
async function waitFor(predicate: () => boolean, timeoutMs = 15000): Promise<void> {
const started = Date.now();
while (!predicate()) {
@ -3904,6 +3918,54 @@ describe("download manager", () => {
expect(completedItems[3].fullStatus).toBe("Fertig (200 MB)");
});
it("clears stale pending extraction labels for untouched items when another archive fails", () => {
const createdAt = Date.now() - 10_000;
const completedItems = [
{
id: "stale-fail-item-1",
status: "completed",
fileName: "show.s01e01.part1.rar",
downloadedBytes: 100 * 1024 * 1024,
fullStatus: "Fertig (100 MB)",
updatedAt: createdAt
},
{
id: "stale-fail-item-2",
status: "completed",
fileName: "show.s01e01.part2.rar",
downloadedBytes: 100 * 1024 * 1024,
fullStatus: "Fertig (100 MB)",
updatedAt: createdAt
},
{
id: "stale-fail-item-3",
status: "completed",
fileName: "show.s01e05.part2.rar",
downloadedBytes: 180 * 1024 * 1024,
fullStatus: "Entpacken - Ausstehend",
updatedAt: createdAt
}
] as any[];
const previousStatuses = new Map<string, string>(completedItems.map((item: any) => [item.id, item.fullStatus]));
completedItems[0].fullStatus = "Entpacken - Error";
completedItems[1].fullStatus = "Entpacken - Error";
(DownloadManager.prototype as any).applyPackageExtractFailureStatuses.call(
{},
completedItems,
(archiveName: string) => resolveArchiveItemsFromList(archiveName, completedItems),
new Map([["show.s01e01.part1.rar", "Checksum error in the encrypted file"]]),
"Checksum error in the encrypted file",
previousStatuses,
createdAt + 5_000
);
expect(completedItems[0].fullStatus).toBe("Entpack-Fehler: Checksum error in the encrypted file");
expect(completedItems[1].fullStatus).toBe("Entpack-Fehler: Checksum error in the encrypted file");
expect(completedItems[2].fullStatus).toBe("Fertig (180 MB)");
});
it("detects start conflicts when extract output already exists", async () => {
const root = fs.mkdtempSync(path.join(os.tmpdir(), "rd-dm-"));
tempDirs.push(root);