feat: module media — upload, miniatures Pillow et suppression

Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>
This commit is contained in:
2026-05-24 05:05:37 +02:00
parent 199565e77c
commit cb85801061
4 changed files with 219 additions and 1 deletions
+30 -1
View File
@@ -1,3 +1,32 @@
from fastapi import APIRouter
from fastapi import APIRouter, UploadFile, File, Query, HTTPException
from fastapi.responses import Response
from app.schemas.media import MediaUploadResponse
from app.services.media import save_image, save_audio, delete_media, ALLOWED_IMAGE_TYPES, ALLOWED_AUDIO_TYPES
router = APIRouter()
@router.post("/upload", response_model=MediaUploadResponse)
async def upload_media(
file: UploadFile = File(...),
context: str = Query(default="note", pattern="^(product|note|attachment)$"),
):
if file.content_type in ALLOWED_IMAGE_TYPES:
result = await save_image(file, context=context)
elif file.content_type in ALLOWED_AUDIO_TYPES:
result = await save_audio(file)
else:
raise HTTPException(status_code=400, detail=f"Type de fichier non supporté : {file.content_type}")
return MediaUploadResponse(**result)
@router.delete("/{file_id}", status_code=204)
async def delete_media_endpoint(
file_id: str,
file_path: str = Query(...),
thumbnail_path: str | None = Query(default=None),
):
delete_media(file_id=file_id, file_path=file_path, thumbnail_path=thumbnail_path)
return Response(status_code=204)
+8
View File
@@ -0,0 +1,8 @@
from pydantic import BaseModel
class MediaUploadResponse(BaseModel):
file_id: str
file_path: str
thumbnail_path: str | None
file_type: str
+85
View File
@@ -0,0 +1,85 @@
import io
import uuid
from pathlib import Path
from fastapi import HTTPException, UploadFile
from PIL import Image
from app.core.config import settings
UPLOAD_DIR: Path = settings.upload_path
ALLOWED_IMAGE_TYPES = {"image/jpeg", "image/png", "image/webp", "image/svg+xml"}
ALLOWED_AUDIO_TYPES = {"audio/webm", "audio/mp4"}
THUMBNAIL_SIZES = {
"product": (150, 150),
"note": (300, 300),
"attachment": (400, 300),
}
async def save_image(file: UploadFile, context: str = "note") -> dict:
if file.content_type not in ALLOWED_IMAGE_TYPES:
raise HTTPException(status_code=400, detail=f"Format non supporté : {file.content_type}")
file_id = str(uuid.uuid4())
content = await file.read()
orig_dir = UPLOAD_DIR / "images" / "originals"
orig_dir.mkdir(parents=True, exist_ok=True)
orig_path = orig_dir / f"{file_id}.webp"
if file.content_type == "image/svg+xml":
orig_path.write_bytes(content)
return {
"file_id": file_id,
"file_path": str(orig_path.relative_to(UPLOAD_DIR)),
"thumbnail_path": None,
"file_type": "image",
}
img = Image.open(io.BytesIO(content)).convert("RGB")
img.save(orig_path, "WEBP", quality=85)
thumb_dir = UPLOAD_DIR / "images" / "thumbnails"
thumb_dir.mkdir(parents=True, exist_ok=True)
thumb_path = thumb_dir / f"{file_id}_thumb.webp"
size = THUMBNAIL_SIZES.get(context, (300, 300))
img_thumb = Image.open(io.BytesIO(content)).convert("RGB")
img_thumb.thumbnail(size, Image.LANCZOS)
img_thumb.save(thumb_path, "WEBP", quality=80)
return {
"file_id": file_id,
"file_path": str(orig_path.relative_to(UPLOAD_DIR)),
"thumbnail_path": str(thumb_path.relative_to(UPLOAD_DIR)),
"file_type": "image",
}
async def save_audio(file: UploadFile) -> dict:
if file.content_type not in ALLOWED_AUDIO_TYPES:
raise HTTPException(status_code=400, detail=f"Format audio non supporté : {file.content_type}")
file_id = str(uuid.uuid4())
audio_dir = UPLOAD_DIR / "audio"
audio_dir.mkdir(parents=True, exist_ok=True)
ext = ".webm" if "webm" in (file.content_type or "") else ".m4a"
audio_path = audio_dir / f"{file_id}{ext}"
audio_path.write_bytes(await file.read())
return {
"file_id": file_id,
"file_path": str(audio_path.relative_to(UPLOAD_DIR)),
"thumbnail_path": None,
"file_type": "audio",
}
def delete_media(file_id: str, file_path: str, thumbnail_path: str | None = None) -> None:
(UPLOAD_DIR / file_path).unlink(missing_ok=True)
if thumbnail_path:
(UPLOAD_DIR / thumbnail_path).unlink(missing_ok=True)
+96
View File
@@ -0,0 +1,96 @@
import io
import pytest
from pathlib import Path
def _make_jpeg(width: int = 200, height: int = 200) -> bytes:
try:
from PIL import Image
img = Image.new("RGB", (width, height), color=(120, 80, 40))
buf = io.BytesIO()
img.save(buf, format="JPEG")
return buf.getvalue()
except ImportError:
pytest.skip("Pillow non disponible localement (Python 3.14) — OK en Docker")
async def test_upload_image_retourne_chemins(client, tmp_path, monkeypatch):
from app.services import media as media_svc
monkeypatch.setattr(media_svc, "UPLOAD_DIR", tmp_path)
response = await client.post(
"/api/media/upload",
files={"file": ("photo.jpg", _make_jpeg(), "image/jpeg")},
params={"context": "note"},
)
assert response.status_code == 200
data = response.json()
assert "file_id" in data
assert data["file_path"].startswith("images/originals/")
assert data["thumbnail_path"].startswith("images/thumbnails/")
async def test_upload_format_invalide_retourne_400(client, tmp_path, monkeypatch):
from app.services import media as media_svc
monkeypatch.setattr(media_svc, "UPLOAD_DIR", tmp_path)
response = await client.post(
"/api/media/upload",
files={"file": ("doc.pdf", b"contenu pdf", "application/pdf")},
)
assert response.status_code == 400
async def test_miniature_produit_max_150px(client, tmp_path, monkeypatch):
try:
from PIL import Image
except ImportError:
pytest.skip("Pillow non disponible localement")
from app.services import media as media_svc
monkeypatch.setattr(media_svc, "UPLOAD_DIR", tmp_path)
response = await client.post(
"/api/media/upload",
files={"file": ("prod.jpg", _make_jpeg(800, 600), "image/jpeg")},
params={"context": "product"},
)
assert response.status_code == 200
data = response.json()
thumb = Image.open(tmp_path / data["thumbnail_path"])
assert thumb.width <= 150
assert thumb.height <= 150
async def test_upload_audio_retourne_chemin(client, tmp_path, monkeypatch):
from app.services import media as media_svc
monkeypatch.setattr(media_svc, "UPLOAD_DIR", tmp_path)
response = await client.post(
"/api/media/upload",
files={"file": ("enreg.webm", b"fake webm bytes", "audio/webm")},
params={"context": "note"},
)
assert response.status_code == 200
data = response.json()
assert data["file_path"].startswith("audio/")
assert data["thumbnail_path"] is None
async def test_suppression_media(client, tmp_path, monkeypatch):
from app.services import media as media_svc
monkeypatch.setattr(media_svc, "UPLOAD_DIR", tmp_path)
up = await client.post(
"/api/media/upload",
files={"file": ("enreg.webm", b"fake webm bytes", "audio/webm")},
params={"context": "note"},
)
data = up.json()
file_id = data["file_id"]
response = await client.delete(
f"/api/media/{file_id}",
params={"file_path": data["file_path"]},
)
assert response.status_code == 204
assert not (tmp_path / data["file_path"]).exists()