|
|
|
|
| import io
|
| import numpy as np
|
| from PIL import Image
|
| from fastapi import FastAPI, File, UploadFile
|
| from fastapi.responses import JSONResponse
|
| from tensorflow.keras.models import load_model
|
|
|
|
|
| app = FastAPI(
|
| title="API de Clasificación de TBC (Mejorado)",
|
| description="Una API para clasificar radiografías de tórax como 'normal' o 'tbc' usando un modelo CNN con Transfer Learning (MobileNetV2).",
|
| version="2.0"
|
| )
|
|
|
|
|
|
|
| try:
|
|
|
| model = load_model('modelo_cnn_tbc_mejorado.h5')
|
| print("Modelo mejorado cargado exitosamente.")
|
| except Exception as e:
|
| print(f"Error al cargar el modelo: {e}")
|
| model = None
|
|
|
|
|
| IMG_HEIGHT = 224
|
| IMG_WIDTH = 224
|
| CLASS_NAMES = ["normal", "tbc"]
|
|
|
|
|
|
|
| @app.get("/")
|
| def read_root():
|
| """Endpoint raíz para verificar que la API está funcionando."""
|
| return {"message": "Bienvenido a la API de Clasificación de TBC v2.0. Usa el endpoint /predict/."}
|
|
|
| @app.post("/predict/")
|
| async def predict(file: UploadFile = File(...)):
|
| """
|
| Endpoint para predecir si una radiografía es 'normal' o 'tbc'.
|
| - Acepta un archivo de imagen (JPG, PNG, etc.).
|
| - Devuelve la clase predicha y la confianza de la predicción.
|
| """
|
| if not model:
|
| return JSONResponse(status_code=500, content={"error": "El modelo no está cargado. Revisa los logs del servidor."})
|
|
|
|
|
| contents = await file.read()
|
|
|
|
|
| try:
|
| image = Image.open(io.BytesIO(contents)).convert('RGB')
|
| except Exception as e:
|
| return JSONResponse(status_code=400, content={"error": f"Archivo inválido. No se pudo procesar la imagen: {e}"})
|
|
|
|
|
|
|
| image = image.resize((IMG_WIDTH, IMG_HEIGHT))
|
|
|
| img_array = np.array(image)
|
|
|
| img_array = img_array / 255.0
|
|
|
| image_batch = np.expand_dims(img_array, axis=0)
|
|
|
|
|
| prediction = model.predict(image_batch)
|
| score = prediction[0][0]
|
|
|
|
|
| if score > 0.5:
|
| predicted_class = CLASS_NAMES[1]
|
| confidence = score
|
| else:
|
| predicted_class = CLASS_NAMES[0]
|
| confidence = 1 - score
|
|
|
|
|
| return {
|
| "filename": file.filename,
|
| "prediction": predicted_class,
|
| "confidence": float(confidence)
|
| } |