MilkSpoilageClassifier-API / fastapi_app.py
chenhaoq87's picture
Update description to match Gradio microbial analysis theme
d5c2352 verified
"""
Simple FastAPI REST API for Milk Spoilage Classification
This provides a clean REST endpoint for Custom GPT and other integrations.
"""
from fastapi import FastAPI
from fastapi.middleware.cors import CORSMiddleware
from pydantic import BaseModel, Field
import joblib
import numpy as np
from typing import Dict
# Load model
model = joblib.load("model.joblib")
# Create FastAPI app
app = FastAPI(
title="Milk Spoilage Classification API",
description="AI-powered spoilage classification system for predicting milk spoilage type based on bacterial contamination and microbial count analysis",
version="1.0.0"
)
# Add CORS middleware - Completely open for Custom GPT and all origins
app.add_middleware(
CORSMiddleware,
allow_origins=["*"],
allow_credentials=False,
allow_methods=["*"],
allow_headers=["*"],
max_age=3600,
)
# Request/Response models
class PredictionInput(BaseModel):
spc_d7: float = Field(..., description="Standard Plate Count at Day 7 (log CFU/mL, base 10)", ge=0.0, le=10.0)
spc_d14: float = Field(..., description="Standard Plate Count at Day 14 (log CFU/mL, base 10)", ge=0.0, le=10.0)
spc_d21: float = Field(..., description="Standard Plate Count at Day 21 (log CFU/mL, base 10)", ge=0.0, le=10.0)
tgn_d7: float = Field(..., description="Total Gram-Negative at Day 7 (log CFU/mL, base 10)", ge=0.0, le=10.0)
tgn_d14: float = Field(..., description="Total Gram-Negative at Day 14 (log CFU/mL, base 10)", ge=0.0, le=10.0)
tgn_d21: float = Field(..., description="Total Gram-Negative at Day 21 (log CFU/mL, base 10)", ge=0.0, le=10.0)
class Config:
json_schema_extra = {
"example": {
"spc_d7": 2.1,
"spc_d14": 4.7,
"spc_d21": 6.4,
"tgn_d7": 1.0,
"tgn_d14": 3.7,
"tgn_d21": 5.3
}
}
class PredictionOutput(BaseModel):
prediction: str = Field(..., description="Predicted spoilage class")
probabilities: Dict[str, float] = Field(..., description="Probability for each class")
confidence: float = Field(..., description="Confidence score (max probability)")
@app.get("/")
async def root():
"""Root endpoint with API information."""
return {
"message": "Milk Spoilage Classification API",
"endpoints": {
"predict": "/predict",
"health": "/health",
"docs": "/docs"
}
}
@app.post("/predict", response_model=PredictionOutput, tags=["Prediction"])
async def predict(input_data: PredictionInput):
"""
Predict milk spoilage type based on microbial counts.
Accepts log CFU/mL values (base 10) and converts to raw CFU/mL for the model.
Returns the predicted class, probabilities for all classes, and confidence score.
"""
# Convert log values to raw CFU/mL
raw_spc_d7 = 10 ** input_data.spc_d7
raw_spc_d14 = 10 ** input_data.spc_d14
raw_spc_d21 = 10 ** input_data.spc_d21
raw_tgn_d7 = 10 ** input_data.tgn_d7
raw_tgn_d14 = 10 ** input_data.tgn_d14
raw_tgn_d21 = 10 ** input_data.tgn_d21
# Prepare features with raw values
features = np.array([[
raw_spc_d7, raw_spc_d14, raw_spc_d21,
raw_tgn_d7, raw_tgn_d14, raw_tgn_d21
]])
# Make prediction
prediction = model.predict(features)[0]
probabilities = model.predict_proba(features)[0]
# Format response
prob_dict = {
str(cls): float(prob)
for cls, prob in zip(model.classes_, probabilities)
}
return PredictionOutput(
prediction=str(prediction),
probabilities=prob_dict,
confidence=float(max(probabilities))
)
@app.get("/health", tags=["Health"])
async def health_check():
"""Health check endpoint."""
return {
"status": "healthy",
"model_loaded": model is not None,
"classes": model.classes_.tolist()
}
if __name__ == "__main__":
import uvicorn
uvicorn.run(app, host="0.0.0.0", port=7860)