feat(model-manager): add "Delete all" gallery button + clearer 401 errors

- Gallery: DELETE /api/gallery/all removes every image under output/;
  "Delete all" button with in-app confirm and a deleted/failed count.
- Downloads: surface a clear, actionable message when CivitAI/HuggingFace
  returns 401/403 (model requires login/early-access, or the key/token
  lacks access) instead of a bare error, both at resolve time and during
  the download stream.

Co-Authored-By: Claude Opus 4.8 <noreply@anthropic.com>
This commit is contained in:
2026-06-07 16:19:48 -04:00
parent b919e55206
commit 6871a6d460
5 changed files with 55 additions and 3 deletions
+5 -2
View File
@@ -111,8 +111,11 @@ async def run_download(download_id: int, url: str, headers: dict[str, str],
db.update_download(download_id, status="canceled", error="Canceled by user")
_cleanup(part_path)
except httpx.HTTPStatusError as exc:
db.update_download(download_id, status="failed",
error=f"HTTP {exc.response.status_code}")
code = exc.response.status_code
msg = f"HTTP {code}"
if code in (401, 403):
msg += " — requires login/early-access or invalid API key"
db.update_download(download_id, status="failed", error=msg)
_cleanup(part_path)
except Exception as exc: # noqa: BLE001 - surface any failure to the UI
db.update_download(download_id, status="failed", error=str(exc))
+26
View File
@@ -6,6 +6,7 @@ import os
from pathlib import Path
from typing import Optional
import httpx
from fastapi import FastAPI, HTTPException
from fastapi.responses import FileResponse
from fastapi.staticfiles import StaticFiles
@@ -141,6 +142,15 @@ async def start_download(body: DownloadIn) -> dict:
try:
resolved = await registries.resolve(url)
except httpx.HTTPStatusError as exc:
code = exc.response.status_code
if code in (401, 403):
raise HTTPException(
400,
f"The source returned {code} for this model. It likely requires "
"being logged in / early access on that site, or your API key/token "
"doesn't have access to it. (Public models download fine.)")
raise HTTPException(400, f"Could not resolve URL (HTTP {code})")
except Exception as exc: # noqa: BLE001 - report resolution failure to the user
raise HTTPException(400, f"Could not resolve URL: {exc}")
@@ -246,6 +256,22 @@ def list_gallery(limit: int = 60, offset: int = 0) -> dict:
"offset": offset, "returned": len(page)}
@app.delete("/api/gallery/all")
def delete_all_photos() -> dict:
"""Permanently delete every image under OUTPUT_DIR."""
deleted = 0
failed = 0
if OUTPUT_DIR.is_dir():
for path in OUTPUT_DIR.rglob("*"):
if path.is_file() and path.suffix.lower() in IMAGE_EXTS:
try:
path.unlink()
deleted += 1
except OSError:
failed += 1
return {"deleted": deleted, "failed": failed}
@app.get("/gallery/file")
def gallery_file(path: str):
try:
+21
View File
@@ -376,7 +376,28 @@ document.addEventListener("keydown", (e) => {
$("#lbDelete").addEventListener("click", () => {
if (lbCurrent) deletePhoto(lbCurrent.p, lbCurrent.card);
});
async function deleteAllPhotos() {
if (!galleryState.total) {
toast($("#galleryToast"), "No photos to delete", false);
return;
}
const ok = await confirmDialog(
`Permanently delete ALL ${galleryState.total} photo(s)? This cannot be undone.`,
"Delete all");
if (!ok) return;
try {
const res = await api("/api/gallery/all", { method: "DELETE" });
let msg = `Deleted ${res.deleted} photo(s)`;
if (res.failed) msg += ` (${res.failed} could not be removed — permissions)`;
toast($("#galleryToast"), msg, res.failed > 0);
loadGallery(true);
} catch (err) {
toast($("#galleryToast"), err.message, true);
}
}
$("#refreshGallery").addEventListener("click", () => loadGallery(true));
$("#deleteAllGallery").addEventListener("click", deleteAllPhotos);
$("#galleryMore").addEventListener("click", () => loadGallery(false));
// ---- browse civitai -------------------------------------------------------
+1
View File
@@ -58,6 +58,7 @@
<div class="head-actions">
<span class="form-msg" id="galleryToast"></span>
<button class="btn" id="refreshGallery">Refresh</button>
<button class="btn danger" id="deleteAllGallery">Delete all</button>
</div>
</header>
<div id="galleryGrid" class="gallery-grid">