- Fix PPTX/PDF export: Puppeteer URL port mismatch (80 → 3000) - Fix backend export_utils to use NEXT_INTERNAL_URL env var - Add Chromium to frontend Dockerfile for Docker-based export - Fix slide edit socket hang up with asyncio.wait_for() timeouts - Add FastAPI StaticFiles mounts for /static and /app_data - Add Next.js rewrite for /static/ to proxy to backend - Show template thumbnail in master decks admin page - Add error logging to ReviewWorkflow component - Add Docker env vars for web service (APP_DATA_DIRECTORY, app_data volume) - Add project README in English Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>
112 lines
3.7 KiB
Python
112 lines
3.7 KiB
Python
import asyncio
|
|
import traceback
|
|
from typing import Annotated, Optional
|
|
from fastapi import APIRouter, Body, Depends, HTTPException
|
|
from sqlalchemy.ext.asyncio import AsyncSession
|
|
import uuid
|
|
|
|
from models.sql.presentation import PresentationModel
|
|
from models.sql.slide import SlideModel
|
|
from services.database import get_async_session
|
|
from services.image_generation_service import ImageGenerationService
|
|
from utils.asset_directory_utils import get_images_directory
|
|
from utils.llm_calls.edit_slide import get_edited_slide_content
|
|
from utils.llm_calls.edit_slide_html import get_edited_slide_html
|
|
from utils.llm_calls.select_slide_type_on_edit import get_slide_layout_from_prompt
|
|
from utils.process_slides import process_old_and_new_slides_and_fetch_assets
|
|
|
|
|
|
SLIDE_ROUTER = APIRouter(prefix="/slide", tags=["Slide"])
|
|
|
|
|
|
@SLIDE_ROUTER.post("/edit")
|
|
async def edit_slide(
|
|
id: Annotated[uuid.UUID, Body()],
|
|
prompt: Annotated[str, Body()],
|
|
sql_session: AsyncSession = Depends(get_async_session),
|
|
):
|
|
slide = await sql_session.get(SlideModel, id)
|
|
if not slide:
|
|
raise HTTPException(status_code=404, detail="Slide not found")
|
|
presentation = await sql_session.get(PresentationModel, slide.presentation)
|
|
if not presentation:
|
|
raise HTTPException(status_code=404, detail="Presentation not found")
|
|
|
|
try:
|
|
presentation_layout = presentation.get_layout()
|
|
slide_layout = await asyncio.wait_for(
|
|
get_slide_layout_from_prompt(prompt, presentation_layout, slide),
|
|
timeout=60,
|
|
)
|
|
|
|
edited_slide_content = await asyncio.wait_for(
|
|
get_edited_slide_content(
|
|
prompt, slide, presentation.language, slide_layout
|
|
),
|
|
timeout=90,
|
|
)
|
|
|
|
image_generation_service = ImageGenerationService(get_images_directory())
|
|
|
|
# This will mutate edited_slide_content
|
|
new_assets = await asyncio.wait_for(
|
|
process_old_and_new_slides_and_fetch_assets(
|
|
image_generation_service,
|
|
slide.content,
|
|
edited_slide_content,
|
|
),
|
|
timeout=120,
|
|
)
|
|
except asyncio.TimeoutError:
|
|
raise HTTPException(
|
|
status_code=504,
|
|
detail="Slide editing timed out. The AI model or image generation took too long. Please try again.",
|
|
)
|
|
except HTTPException:
|
|
raise
|
|
except Exception as e:
|
|
traceback.print_exc()
|
|
raise HTTPException(
|
|
status_code=500,
|
|
detail=f"Failed to edit slide: {str(e)}",
|
|
)
|
|
|
|
# Always assign a new unique id to the slide
|
|
slide.id = uuid.uuid4()
|
|
|
|
sql_session.add(slide)
|
|
slide.content = edited_slide_content
|
|
slide.layout = slide_layout.id
|
|
slide.speaker_note = edited_slide_content.get("__speaker_note__", "")
|
|
sql_session.add_all(new_assets)
|
|
await sql_session.commit()
|
|
|
|
return slide
|
|
|
|
|
|
@SLIDE_ROUTER.post("/edit-html", response_model=SlideModel)
|
|
async def edit_slide_html(
|
|
id: Annotated[uuid.UUID, Body()],
|
|
prompt: Annotated[str, Body()],
|
|
html: Annotated[Optional[str], Body()] = None,
|
|
sql_session: AsyncSession = Depends(get_async_session),
|
|
):
|
|
slide = await sql_session.get(SlideModel, id)
|
|
if not slide:
|
|
raise HTTPException(status_code=404, detail="Slide not found")
|
|
|
|
html_to_edit = html or slide.html_content
|
|
if not html_to_edit:
|
|
raise HTTPException(status_code=400, detail="No HTML to edit")
|
|
|
|
edited_slide_html = await get_edited_slide_html(prompt, html_to_edit)
|
|
|
|
# Always assign a new unique id to the slide
|
|
# This is to ensure that the nextjs can track slide updates
|
|
slide.id = uuid.uuid4()
|
|
|
|
sql_session.add(slide)
|
|
slide.html_content = edited_slide_html
|
|
await sql_session.commit()
|
|
|
|
return slide
|