feat(admin): add "Regenerate All GLB + USD" button

New endpoint POST /admin/settings/regenerate-all-canonical-scenes
queues GLB + USD master export for ALL completed CAD files, replacing
existing assets. Used after pipeline changes that affect tessellation
or normals.

Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>
This commit is contained in:
2026-03-13 15:21:04 +01:00
parent 253f11a945
commit 5afe502bc0
2 changed files with 40 additions and 0 deletions
+22
View File
@@ -607,6 +607,28 @@ async def generate_missing_usd_masters(
return {"queued": queued, "message": f"Queued {queued} missing USD master task(s)"}
@router.post("/settings/regenerate-all-canonical-scenes", status_code=status.HTTP_202_ACCEPTED)
async def regenerate_all_canonical_scenes(
admin: User = Depends(require_global_admin),
db: AsyncSession = Depends(get_db),
):
"""Re-queue GLB + USD master export for ALL completed CAD files (overwrites existing assets)."""
result = await db.execute(
select(CadFile).where(CadFile.processing_status == ProcessingStatus.completed)
)
cad_files = result.scalars().all()
from app.tasks.step_tasks import generate_gltf_geometry_task
queued = 0
for cad_file in cad_files:
if not cad_file.stored_path:
continue
generate_gltf_geometry_task.delay(str(cad_file.id))
queued += 1
return {"queued": queued, "message": f"Queued {queued} canonical scene regeneration task(s)"}
@router.post("/settings/recover-stuck-processing", status_code=status.HTTP_200_OK)
async def recover_stuck_processing(
admin: User = Depends(require_global_admin),
+18
View File
@@ -237,6 +237,12 @@ export default function AdminPage() {
onError: (e: any) => toast.error(e.response?.data?.detail || 'Failed'),
})
const regenerateAllCanonicalScenesMut = useMutation({
mutationFn: () => api.post('/admin/settings/regenerate-all-canonical-scenes'),
onSuccess: (res) => toast.success(res.data.message || 'All canonical scenes re-queued'),
onError: (e: any) => toast.error(e.response?.data?.detail || 'Failed'),
})
const [smtpDraft, setSmtpDraft] = useState<Partial<Settings>>({})
const smtp = { ...settings, ...smtpDraft } as Settings
@@ -958,6 +964,18 @@ export default function AdminPage() {
</button>
<p className="text-xs text-content-muted">Queues geometry GLB + USD master for all completed CAD files missing a canonical scene.</p>
</div>
<div className="flex flex-col gap-1">
<button
onClick={() => regenerateAllCanonicalScenesMut.mutate()}
disabled={regenerateAllCanonicalScenesMut.isPending}
className="btn-secondary text-sm w-full justify-start"
title="Re-export geometry GLB + USD master for ALL completed CAD files (overwrites existing)"
>
<RefreshCw size={14} className={regenerateAllCanonicalScenesMut.isPending ? 'animate-spin' : ''} />
{regenerateAllCanonicalScenesMut.isPending ? 'Queueing…' : 'Regenerate All GLB + USD'}
</button>
<p className="text-xs text-content-muted">Re-exports GLB and USD for all completed CAD files, replacing existing assets.</p>
</div>
<div className="flex flex-col gap-1">
<button
onClick={() => importMediaAssetsMut.mutate()}