Files
xian_algorithm_new/app/api/rainfall.py
T

160 lines
6.1 KiB
Python
Raw Normal View History

2026-06-06 08:38:19 +08:00
"""
暴雨灾害链预测接口
"""
import asyncio
2026-06-06 13:18:25 +08:00
from datetime import datetime
2026-06-06 08:38:19 +08:00
from typing import List, Dict, Any, Optional
from fastapi import APIRouter, HTTPException
2026-06-12 16:09:57 +08:00
from app.schemas.api_schemas import RainfallPredictRequest, PredictResponse, PredictionItem, UpdateMonitoringTimeRequest
2026-06-06 08:38:19 +08:00
from app.utils.api_deps import get_rainfall_model, get_prediction_semaphore
from app.repositories.dbn_repository import dbn_repository
2026-06-12 16:09:57 +08:00
from app.core.rainfall_manager import rainfall_manager
2026-06-06 08:38:19 +08:00
from app.config.paths import get_logger
2026-06-12 16:09:57 +08:00
from app.utils.time_converter import TimeConverter
2026-06-06 08:38:19 +08:00
router = APIRouter(prefix="/rainfall", tags=["暴雨灾害链"])
logger = get_logger("api.rainfall")
SOURCE_TYPE_MAP = {1: "隐患点", 2: "风险点"}
LEVEL_MAP = {"": "", "": "", "较高": "较高", "": ""}
2026-06-14 15:31:42 +08:00
def _build_prediction_map(results: List[Dict[str, Any]]) -> Dict[str, float]:
"""将模型原始结果转换为存储格式: {id_type: 概率百分比}"""
result_map = {}
2026-06-06 08:38:19 +08:00
for r in results:
probs = r.get("disaster_probabilities", {})
if not probs:
continue
2026-06-14 15:31:42 +08:00
source_id = r["source_id"]
source_type = r.get("source_type")
2026-06-06 08:38:19 +08:00
max_hazard = max(probs, key=probs.get)
2026-06-14 15:31:42 +08:00
# key 格式: {source_id}_{source_type}value 为百分比概率
key = f"{source_id}_{source_type}"
result_map[key] = round(probs[max_hazard] * 100, 2)
return result_map
2026-06-06 08:38:19 +08:00
def _fetch_points(point_ids: Optional[List[int]], region_code: Optional[str]) -> List[Dict[str, Any]]:
"""获取点位列表"""
if point_ids:
return dbn_repository.get_points_by_ids(point_ids)
return dbn_repository.get_all_points(region_code)
def _predict_sync(point_ids: Optional[List[int]], region_code: Optional[str],
2026-06-06 13:18:25 +08:00
rainfall: Optional[float], duration: Optional[float],
operation_type: str) -> tuple:
"""
同步执行暴雨预测(在线程池中运行)
Returns:
2026-06-14 15:52:33 +08:00
(结果map, 实际使用的降雨数据, 当前时间)
2026-06-06 13:18:25 +08:00
"""
2026-06-06 08:38:19 +08:00
points = _fetch_points(point_ids, region_code)
if not points:
2026-06-14 15:31:42 +08:00
return {}, {}, datetime.now()
2026-06-06 08:38:19 +08:00
model = get_rainfall_model()
2026-06-06 13:18:25 +08:00
raw_results = model.predict_multiple_points(points, rainfall=rainfall, duration=duration)
2026-06-14 15:31:42 +08:00
result_map = _build_prediction_map(raw_results)
2026-06-06 13:18:25 +08:00
2026-06-14 15:52:33 +08:00
# 获取实际使用的降雨数据(如果未传递,模型会从数据库查询)
actual_rainfall = rainfall
actual_duration = duration
if actual_rainfall is None or actual_duration is None:
# 获取第一个点的降雨数据作为参考
from app.repositories.dbn_repository import DbnRepository
first_point = points[0]
rain_data = DbnRepository.get_rainfall_data_with_duration(first_point['lon'], first_point['lat'])
if actual_rainfall is None:
actual_rainfall = rain_data.get('accum_rain', 0.0)
if actual_duration is None:
actual_duration = rain_data.get('duration_hours', 0)
# 构建经过默认值处理的条件用于保存
2026-06-06 13:18:25 +08:00
now = datetime.now()
condition = {
"point_ids": point_ids,
"region_code": region_code,
2026-06-14 15:52:33 +08:00
"rainfall": actual_rainfall,
"duration": actual_duration
2026-06-06 13:18:25 +08:00
}
2026-06-14 15:31:42 +08:00
return result_map, condition, now
2026-06-06 08:38:19 +08:00
2026-06-12 16:09:57 +08:00
@router.post("/update-monitoring-time", summary="更新降雨监测查询时间")
async def update_monitoring_time(req: UpdateMonitoringTimeRequest):
"""
更新降雨站点监测的查询时间,触发重新计算
- **query_time**: 新的查询时间,格式: YYYY-MM-DD HH:mm:ss
"""
try:
# 将字符串时间解析为 datetime 对象
new_time = TimeConverter.parse_input_time(req.query_time)
# 更新监测时间,触发重新计算
result = rainfall_manager.update_query_time(new_time)
logger.info(f"更新监测时间成功: {result}")
return {
"code": 200,
"message": "success",
"data": result
}
except ValueError as e:
logger.error(f"时间格式错误: {e}")
raise HTTPException(status_code=400, detail=f"时间格式错误: {e}")
except Exception as e:
logger.error(f"更新监测时间失败: {e}", exc_info=True)
raise HTTPException(status_code=500, detail=f"更新监测时间失败: {e}")
2026-06-06 08:38:19 +08:00
@router.post("/predict", response_model=PredictResponse, summary="暴雨灾害链预测")
async def predict_rainfall(req: RainfallPredictRequest):
"""
2026-06-14 15:31:42 +08:00
根据降雨量和持续时间,批量预测隐患点/风险点的灾害概率。
2026-06-06 08:38:19 +08:00
2026-06-14 14:38:20 +08:00
- **disaster_name**: 灾害名称
2026-06-06 08:38:19 +08:00
- **point_ids**: 点位ID列表(可选,不传则查询所有点)
- **region_code**: 行政区划代码(可选,不传则不限区域)
2026-06-06 13:18:25 +08:00
- **rainfall**: 累计降雨量(mm),不传则从气象表自动获取
- **duration**: 降雨持续时间(h),不传则从气象表自动获取
- **operation_type**: 操作类型(如 '实时监测', '情景模拟', '应急评估'
2026-06-06 08:38:19 +08:00
"""
semaphore = get_prediction_semaphore()
async with semaphore:
loop = asyncio.get_event_loop()
try:
2026-06-14 15:31:42 +08:00
result_map, condition, now = await loop.run_in_executor(
2026-06-06 08:38:19 +08:00
None, _predict_sync, req.point_ids, req.region_code,
2026-06-06 13:18:25 +08:00
req.rainfall, req.duration, req.operation_type
2026-06-06 08:38:19 +08:00
)
except Exception as e:
logger.error(f"暴雨预测失败: {e}", exc_info=True)
raise HTTPException(status_code=500, detail=f"预测失败: {e}")
2026-06-06 13:18:25 +08:00
# 保存推理结果
record_id = None
2026-06-14 15:31:42 +08:00
if result_map:
2026-06-06 13:18:25 +08:00
try:
record_id = dbn_repository.save_inference_result(
2026-06-14 14:38:20 +08:00
disaster_name=req.disaster_name,
2026-06-06 13:18:25 +08:00
event_type="rainfall",
occurred_time=now,
operation_type=req.operation_type,
condition=condition,
2026-06-14 15:31:42 +08:00
result=result_map
2026-06-06 13:18:25 +08:00
)
logger.info(f"推理结果已保存,record_id={record_id}")
except Exception as e:
logger.error(f"保存推理结果失败: {e}", exc_info=True)
2026-06-14 15:31:42 +08:00
return PredictResponse(code=200, message="success", data=record_id)