Spaces:
Runtime error
Runtime error
File size: 7,480 Bytes
7a511fb 90dfc4a 7a511fb 90dfc4a 7a511fb 7bd9199 7a511fb 7ce6a8b 7a511fb 7ce6a8b 7a511fb 7ce6a8b 7a511fb 7ce6a8b 7a511fb 7ce6a8b 7a511fb 7ce6a8b 7a511fb a3c0f4d 7a511fb a3c0f4d 7a511fb a3c0f4d 7a511fb a3c0f4d 7a511fb |
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 |
from fastapi import APIRouter, HTTPException, Query, Path
from pydantic import BaseModel, field_validator
from typing import List, Optional
from uuid import UUID
from datetime import datetime
from src.config import logger
from src.models import EvaluationType
from src.services import ProposalEvaluationService
class Evaluation(BaseModel):
id: UUID
proposal_id: UUID
evaluation_type: EvaluationType
ai_score: Optional[float] = None
adjusted_score: Optional[float] = None
created_at: datetime
updated_at: datetime
@field_validator("ai_score", "adjusted_score")
@classmethod
def round_to_two_decimals(cls, v):
if v is not None:
return round(v, 2)
return v
class Response(BaseModel):
status: str
data: Optional[List[Evaluation]] = None
class RequestEvaluation(BaseModel):
evaluation_type: Optional[EvaluationType] = None
ai_score: Optional[float] = None
adjusted_score: Optional[float] = None
class DeleteResponse(BaseModel):
status: str
class CreateEvaluation(BaseModel):
evaluation_type: EvaluationType
ai_score: Optional[float] = None
adjusted_score: Optional[float] = None
class CreateEvaluationRequest(BaseModel):
data: List[CreateEvaluation]
class UpdateEvaluation(BaseModel):
evaluation_type: Optional[EvaluationType] = None
ai_score: Optional[float] = None
adjusted_score: Optional[float] = None
class ProposalEvaluationController:
def __init__(self):
self.service = ProposalEvaluationService
self.router = APIRouter()
self.router.add_api_route(
"/rfps/{rfp_id}/proposals/{proposal_id}/evaluations",
self.get_evaluation_by_proposal_and_rfp_id,
methods=["GET"],
response_model=Response,
tags=["Evaluations by Proposal and RFP ID"],
)
self.router.add_api_route(
"/rfps/{rfp_id}/proposals/{proposal_id}/evaluations/{id}",
self.get_evaluation_by_proposal_and_rfp_id_and_id,
methods=["GET"],
response_model=Response,
tags=["Evaluations by Proposal and RFP ID and ID"],
)
self.router.add_api_route(
"/rfps/{rfp_id}/proposals/{proposal_id}/evaluations",
self.create_evaluation_by_proposal_and_rfp_id,
methods=["POST"],
response_model=Response,
tags=["Evaluations by Proposal and RFP ID"],
)
self.router.add_api_route(
"/rfps/{rfp_id}/proposals/{proposal_id}/evaluations/{id}",
self.update_evaluation_by_proposal_and_rfp_id_and_id,
methods=["PUT"],
response_model=Response,
tags=["Evaluations by Proposal and RFP ID and ID"],
)
self.router.add_api_route(
"/rfps/{rfp_id}/proposals/{proposal_id}/evaluations",
self.delete_evaluation_by_proposal_and_rfp_id,
methods=["DELETE"],
response_model=DeleteResponse,
tags=["Evaluations by Proposal and RFP ID"],
)
self.router.add_api_route(
"/rfps/{rfp_id}/proposals/{proposal_id}/evaluations/{id}",
self.delete_evaluation_by_proposal_and_rfp_id_and_id,
methods=["DELETE"],
response_model=DeleteResponse,
tags=["Evaluations by Proposal and RFP ID and ID"],
)
async def get_evaluation_by_proposal_and_rfp_id(
self,
rfp_id: str = Path(...),
proposal_id: str = Path(...),
evaluation_type: Optional[EvaluationType] = Query(None),
):
try:
async with self.service() as service:
results = await service.get_evaluations(
proposal_id=proposal_id, evaluation_type=evaluation_type
)
return Response(
status="success",
data=[Evaluation(**result) for result in results],
)
except Exception as e:
logger.exception(e)
raise HTTPException(status_code=500, detail="Failed to retrieve proposals.")
async def get_evaluation_by_proposal_and_rfp_id_and_id(
self, rfp_id: str = Path(...), proposal_id: str = Path(...), id: str = Path(...)
):
try:
async with self.service() as service:
result = await service.get_evaluations(proposal_id=proposal_id, id=id)
return Response(
status="success",
data=[Evaluation(**result) for result in result],
)
except Exception as e:
logger.exception(e)
raise HTTPException(status_code=500, detail="Failed to retrieve proposals.")
async def create_evaluation_by_proposal_and_rfp_id(
self,
evaluation: CreateEvaluationRequest,
rfp_id: str = Path(...),
proposal_id: str = Path(...),
):
try:
async with self.service() as service:
result = await service.create_evaluations(
proposal_id=proposal_id,
evaluations=evaluation.model_dump(mode="json", exclude_unset=True)[
"data"
],
)
return Response(
status="success",
data=[Evaluation(**result) for result in result],
)
except Exception as e:
logger.exception(e)
raise HTTPException(status_code=500, detail="Failed to retrieve proposals.")
async def update_evaluation_by_proposal_and_rfp_id_and_id(
self,
evaluation: UpdateEvaluation,
rfp_id: str = Path(...),
proposal_id: str = Path(...),
id: str = Path(...),
):
try:
async with self.service() as service:
result = await service.update_evaluations(
proposal_id=proposal_id,
id=id,
evaluations=evaluation.model_dump(mode="json", exclude_unset=True),
)
return Response(
status="success",
data=[Evaluation(**result) for result in result],
)
except Exception as e:
logger.exception(e)
raise HTTPException(status_code=500, detail="Failed to retrieve proposals.")
async def delete_evaluation_by_proposal_and_rfp_id(
self, rfp_id: str = Path(...), proposal_id: str = Path(...)
):
try:
async with self.service() as service:
result = await service.delete_evaluations(proposal_id=proposal_id)
return DeleteResponse(status="success")
except Exception as e:
logger.exception(e)
raise HTTPException(status_code=500, detail="Failed to retrieve proposals.")
async def delete_evaluation_by_proposal_and_rfp_id_and_id(
self, rfp_id: str = Path(...), proposal_id: str = Path(...), id: str = Path(...)
):
try:
async with self.service() as service:
result = await service.delete_evaluations(
proposal_id=proposal_id, id=id
)
return DeleteResponse(status="success")
except Exception as e:
logger.exception(e)
raise HTTPException(status_code=500, detail="Failed to retrieve proposals.")
|