WildOjisan's picture
.
60b4784
import io, os, sys
from typing import List, Dict, Any
from fastapi import FastAPI, UploadFile, File, Query
from fastapi.responses import JSONResponse
from ultralytics import YOLO
from PIL import Image
# ===== 모델 로드 =====
MODEL_NAME = os.getenv("YOLO_MODEL", "/app/models/yolo11n.pt")
if not os.path.isfile(MODEL_NAME):
# 디버그 출력으로 현재 상태 확인
print(">>> YOLO_MODEL =", MODEL_NAME, file=sys.stderr)
try:
print(">>> /app/models =", os.listdir("/app/models"), file=sys.stderr)
except Exception as e:
print(">>> cannot list /app/models:", e, file=sys.stderr)
raise FileNotFoundError(f"weights not found: {MODEL_NAME} (no auto-download)")
model = YOLO(MODEL_NAME) # 여기까지 오면 파일이 확실히 있음
app = FastAPI(title="YOLO FastAPI", version="1.0.0")
@app.get("/health")
def health():
return {"ok": True, "model": MODEL_NAME}
@app.post("/predict")
async def predict(
file: UploadFile = File(...),
conf: float = Query(0.25, ge=0.0, le=1.0, description="confidence threshold"),
iou: float = Query(0.7, ge=0.0, le=1.0, description="IoU threshold (NMS)"),
):
"""
이미지 1장을 받아 YOLO 검출 결과를 JSON으로 반환합니다.
"""
# 1) 이미지 로드
image_bytes = await file.read()
image = Image.open(io.BytesIO(image_bytes)).convert("RGB")
# 2) YOLO 추론
results = model.predict(
source=image,
conf=conf,
iou=iou,
device="cpu",
imgsz=640,
verbose=False,
)
r = results[0]
names = r.names # 클래스 id -> 이름 매핑
out: List[Dict[str, Any]] = []
if r.boxes is not None and len(r.boxes) > 0:
# xyxy, conf, cls를 numpy로
xyxy = r.boxes.xyxy.cpu().numpy()
confs = r.boxes.conf.cpu().numpy()
clses = r.boxes.cls.cpu().numpy().astype(int)
for i in range(len(clses)):
out.append(
{
"class_id": int(clses[i]),
"class_name": names.get(int(clses[i]), str(int(clses[i]))),
"confidence": float(round(confs[i], 4)),
"box": {
"x1": float(xyxy[i][0]),
"y1": float(xyxy[i][1]),
"x2": float(xyxy[i][2]),
"y2": float(xyxy[i][3]),
},
}
)
return JSONResponse(
{
"model": MODEL_NAME,
"num_detections": len(out),
"detections": out,
}
)