2026-04-25 20:43:37 +12:00
|
|
|
from fastapi import APIRouter, Depends, HTTPException, status
|
|
|
|
|
from sqlalchemy import select
|
|
|
|
|
from sqlalchemy.orm import Session
|
|
|
|
|
|
2026-04-25 22:51:36 +12:00
|
|
|
from app.api.deps import AuthSession, require_client_session
|
2026-04-25 20:43:37 +12:00
|
|
|
from app.db.session import get_db
|
|
|
|
|
from app.models.scenario import CostingResult, Scenario
|
|
|
|
|
from app.schemas.scenario import ScenarioCreate, ScenarioRead, ScenarioRunResponse
|
|
|
|
|
from app.services.scenario_engine import run_scenario
|
|
|
|
|
|
|
|
|
|
router = APIRouter(prefix="/api/scenarios", tags=["scenarios"])
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
@router.get("", response_model=list[ScenarioRead])
|
2026-04-25 22:51:36 +12:00
|
|
|
def list_scenarios(session: AuthSession = Depends(require_client_session), db: Session = Depends(get_db)):
|
|
|
|
|
return db.scalars(select(Scenario).where(Scenario.tenant_id == session.tenant_id).order_by(Scenario.created_at.desc())).all()
|
2026-04-25 20:43:37 +12:00
|
|
|
|
|
|
|
|
|
|
|
|
|
@router.post("", response_model=ScenarioRead, status_code=status.HTTP_201_CREATED)
|
2026-04-25 22:51:36 +12:00
|
|
|
def create_scenario(payload: ScenarioCreate, session: AuthSession = Depends(require_client_session), db: Session = Depends(get_db)):
|
|
|
|
|
scenario = Scenario(tenant_id=session.tenant_id, name=payload.name, description=payload.description, overrides=payload.overrides)
|
2026-04-25 20:43:37 +12:00
|
|
|
db.add(scenario)
|
|
|
|
|
db.commit()
|
|
|
|
|
db.refresh(scenario)
|
|
|
|
|
return scenario
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
@router.get("/{scenario_id}", response_model=ScenarioRead)
|
2026-04-25 22:51:36 +12:00
|
|
|
def get_scenario(scenario_id: int, session: AuthSession = Depends(require_client_session), db: Session = Depends(get_db)):
|
|
|
|
|
scenario = db.scalar(select(Scenario).where(Scenario.id == scenario_id, Scenario.tenant_id == session.tenant_id))
|
2026-04-25 20:43:37 +12:00
|
|
|
if scenario is None:
|
|
|
|
|
raise HTTPException(status_code=404, detail="Scenario not found")
|
|
|
|
|
return scenario
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
@router.post("/{scenario_id}/run", response_model=ScenarioRunResponse)
|
2026-04-25 22:51:36 +12:00
|
|
|
def run_scenario_endpoint(scenario_id: int, session: AuthSession = Depends(require_client_session), db: Session = Depends(get_db)):
|
|
|
|
|
scenario = db.scalar(select(Scenario).where(Scenario.id == scenario_id, Scenario.tenant_id == session.tenant_id))
|
2026-04-25 20:43:37 +12:00
|
|
|
if scenario is None:
|
|
|
|
|
raise HTTPException(status_code=404, detail="Scenario not found")
|
|
|
|
|
results = run_scenario(db, scenario)
|
|
|
|
|
db.refresh(scenario)
|
|
|
|
|
return {"scenario": scenario, "results": results}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
@router.get("/{scenario_id}/results")
|
2026-04-25 22:51:36 +12:00
|
|
|
def get_scenario_results(scenario_id: int, session: AuthSession = Depends(require_client_session), db: Session = Depends(get_db)):
|
|
|
|
|
scenario = db.scalar(select(Scenario).where(Scenario.id == scenario_id, Scenario.tenant_id == session.tenant_id))
|
2026-04-25 20:43:37 +12:00
|
|
|
if scenario is None:
|
|
|
|
|
raise HTTPException(status_code=404, detail="Scenario not found")
|
2026-04-25 22:51:36 +12:00
|
|
|
results = db.scalars(
|
|
|
|
|
select(CostingResult).where(CostingResult.scenario_id == scenario_id, CostingResult.tenant_id == session.tenant_id)
|
|
|
|
|
).all()
|
2026-04-25 20:43:37 +12:00
|
|
|
return [
|
|
|
|
|
{
|
|
|
|
|
"product_id": result.product_id,
|
|
|
|
|
"finished_product_delivered": result.finished_product_delivered,
|
|
|
|
|
"distributor_price": result.distributor_price,
|
|
|
|
|
"wholesale_price": result.wholesale_price,
|
|
|
|
|
"warnings": result.warnings,
|
|
|
|
|
"details": result.details,
|
|
|
|
|
}
|
|
|
|
|
for result in results
|
|
|
|
|
]
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
@router.post("/{scenario_id}/approve", response_model=ScenarioRead)
|
2026-04-25 22:51:36 +12:00
|
|
|
def approve_scenario(scenario_id: int, session: AuthSession = Depends(require_client_session), db: Session = Depends(get_db)):
|
|
|
|
|
scenario = db.scalar(select(Scenario).where(Scenario.id == scenario_id, Scenario.tenant_id == session.tenant_id))
|
2026-04-25 20:43:37 +12:00
|
|
|
if scenario is None:
|
|
|
|
|
raise HTTPException(status_code=404, detail="Scenario not found")
|
|
|
|
|
scenario.status = "approved"
|
|
|
|
|
db.commit()
|
|
|
|
|
db.refresh(scenario)
|
|
|
|
|
return scenario
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
@router.post("/{scenario_id}/reject", response_model=ScenarioRead)
|
2026-04-25 22:51:36 +12:00
|
|
|
def reject_scenario(scenario_id: int, session: AuthSession = Depends(require_client_session), db: Session = Depends(get_db)):
|
|
|
|
|
scenario = db.scalar(select(Scenario).where(Scenario.id == scenario_id, Scenario.tenant_id == session.tenant_id))
|
2026-04-25 20:43:37 +12:00
|
|
|
if scenario is None:
|
|
|
|
|
raise HTTPException(status_code=404, detail="Scenario not found")
|
|
|
|
|
scenario.status = "rejected"
|
|
|
|
|
db.commit()
|
|
|
|
|
db.refresh(scenario)
|
|
|
|
|
return scenario
|