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:
@@ -607,6 +607,28 @@ async def generate_missing_usd_masters(
|
|||||||
return {"queued": queued, "message": f"Queued {queued} missing USD master task(s)"}
|
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)
|
@router.post("/settings/recover-stuck-processing", status_code=status.HTTP_200_OK)
|
||||||
async def recover_stuck_processing(
|
async def recover_stuck_processing(
|
||||||
admin: User = Depends(require_global_admin),
|
admin: User = Depends(require_global_admin),
|
||||||
|
|||||||
@@ -237,6 +237,12 @@ export default function AdminPage() {
|
|||||||
onError: (e: any) => toast.error(e.response?.data?.detail || 'Failed'),
|
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 [smtpDraft, setSmtpDraft] = useState<Partial<Settings>>({})
|
||||||
const smtp = { ...settings, ...smtpDraft } as Settings
|
const smtp = { ...settings, ...smtpDraft } as Settings
|
||||||
|
|
||||||
@@ -958,6 +964,18 @@ export default function AdminPage() {
|
|||||||
</button>
|
</button>
|
||||||
<p className="text-xs text-content-muted">Queues geometry GLB + USD master for all completed CAD files missing a canonical scene.</p>
|
<p className="text-xs text-content-muted">Queues geometry GLB + USD master for all completed CAD files missing a canonical scene.</p>
|
||||||
</div>
|
</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">
|
<div className="flex flex-col gap-1">
|
||||||
<button
|
<button
|
||||||
onClick={() => importMediaAssetsMut.mutate()}
|
onClick={() => importMediaAssetsMut.mutate()}
|
||||||
|
|||||||
Reference in New Issue
Block a user