Spaces:
Sleeping
Sleeping
File size: 8,305 Bytes
954be92 1891530 954be92 | 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 211 212 213 214 215 216 217 218 219 220 221 222 223 224 225 226 227 228 229 230 231 232 233 234 235 236 237 238 239 240 241 242 243 244 245 246 247 248 249 250 251 252 253 254 255 256 257 258 259 260 261 262 263 264 265 266 267 268 269 270 271 272 273 274 275 276 277 278 279 280 281 282 283 284 285 286 287 288 289 290 291 292 293 294 295 296 297 | """
API 路由定义
提供汇率查询和换算接口
"""
from fastapi import APIRouter, HTTPException, Request
from typing import Optional
from app.models import (
RatesResponse,
CurrencyRateResponse,
ConvertRequest,
ConvertResponse,
BatchConvertRequest,
BatchConvertResponse,
CurrenciesResponse,
CurrencyInfo,
ServiceStatusResponse
)
from app.utils.currency_names import CURRENCY_INFO, get_currency_info
from app.utils.logger import logger
router = APIRouter(prefix="/api", tags=["汇率接口"])
def get_exchange_service(request: Request):
"""从应用状态获取 ExchangeRateService 实例"""
return request.app.state.exchange_service
# ==================== 获取所有汇率 ====================
@router.get("/rates", response_model=RatesResponse)
async def get_all_rates(request: Request):
"""
获取所有货币汇率
返回缓存中的所有汇率数据,包括基准货币、汇率列表、更新时间等。
"""
service = get_exchange_service(request)
cached = service.get_cached_rates()
if not cached:
logger.warning("API /rates: 没有可用的缓存数据")
raise HTTPException(
status_code=503,
detail="汇率数据不可用,请稍后重试。"
)
return RatesResponse(
success=True,
base_currency=cached.base_code,
rates=cached.rates,
last_update=cached.last_update_utc,
cached_at=cached.cached_at.isoformat(),
currencies_count=len(cached.rates)
)
# ==================== 获取特定货币汇率 ====================
@router.get("/rates/{currency}", response_model=CurrencyRateResponse)
async def get_currency_rate(currency: str, request: Request):
"""
获取特定货币对基准货币的汇率
- **currency**: 货币代码(如 USD, EUR, JPY)
"""
service = get_exchange_service(request)
cached = service.get_cached_rates()
if not cached:
raise HTTPException(
status_code=503,
detail="汇率数据不可用"
)
currency = currency.upper()
if currency not in cached.rates:
raise HTTPException(
status_code=404,
detail=f"未找到货币 '{currency}'"
)
return CurrencyRateResponse(
success=True,
base_currency=cached.base_code,
currency=currency,
rate=cached.rates[currency]
)
# ==================== 两种货币换算 ====================
@router.post("/convert", response_model=ConvertResponse)
async def convert_currency(request: Request, body: ConvertRequest):
"""
两种货币之间的汇率换算
- **from_currency**: 源货币代码
- **to_currency**: 目标货币代码
- **amount**: 换算金额
返回换算结果、汇率和反向汇率。
"""
service = get_exchange_service(request)
result = service.convert(
body.from_currency,
body.to_currency,
body.amount
)
if not result:
raise HTTPException(
status_code=400,
detail="换算失败,请检查货币代码是否有效。"
)
return result
# ==================== GET 方式的货币换算 ====================
@router.get("/convert/{from_currency}/{to_currency}")
async def convert_currency_get(
from_currency: str,
to_currency: str,
amount: float,
request: Request
):
"""
两种货币之间的汇率换算(GET 方式)
- **from_currency**: 源货币代码
- **to_currency**: 目标货币代码
- **amount**: 换算金额(查询参数)
示例: GET /api/convert/USD/CNY?amount=100
"""
service = get_exchange_service(request)
result = service.convert(from_currency, to_currency, amount)
if not result:
raise HTTPException(
status_code=400,
detail="换算失败,请检查货币代码是否有效。"
)
return result
# ==================== 批量换算 ====================
@router.post("/batch-convert", response_model=BatchConvertResponse)
async def batch_convert(request: Request, body: BatchConvertRequest):
"""
批量换算:计算某金额对应所有货币的值
- **base_currency**: 基准货币代码
- **amount**: 换算金额
返回所有货币的换算结果。
"""
service = get_exchange_service(request)
result = service.batch_convert(body.base_currency, body.amount)
if not result:
raise HTTPException(
status_code=400,
detail="批量换算失败,请检查基准货币是否有效。"
)
return result
# ==================== GET 方式的批量换算 ====================
@router.get("/batch-convert/{base_currency}")
async def batch_convert_get(base_currency: str, amount: float, request: Request):
"""
批量换算(GET 方式)
- **base_currency**: 基准货币代码
- **amount**: 换算金额(查询参数)
示例: GET /api/batch-convert/USD?amount=100
"""
service = get_exchange_service(request)
result = service.batch_convert(base_currency, amount)
if not result:
raise HTTPException(
status_code=400,
detail="批量换算失败,请检查基准货币是否有效。"
)
return result
# ==================== 获取货币列表 ====================
@router.get("/currencies", response_model=CurrenciesResponse)
async def get_currencies(request: Request):
"""
获取支持的货币列表(含中英文名称)
返回所有可用货币的代码、名称和符号,按优先级排序。
"""
service = get_exchange_service(request)
settings = service.settings
cached = service.get_cached_rates()
if not cached:
raise HTTPException(
status_code=503,
detail="汇率数据不可用"
)
currencies = []
for code in cached.rates.keys():
info = get_currency_info(code)
currencies.append(CurrencyInfo(
code=code,
name=info.get("name", code),
name_cn=info.get("name_cn", code),
symbol=info.get("symbol", "")
))
# 按优先级排序
priority = settings.PRIORITY_CURRENCIES
currencies.sort(key=lambda x: (
priority.index(x.code) if x.code in priority else 999,
x.code
))
return CurrenciesResponse(
success=True,
currencies=currencies
)
# ==================== 服务状态 ====================
@router.get("/status", response_model=ServiceStatusResponse)
async def get_status(request: Request):
"""
获取服务状态
返回缓存状态、上次更新时间、货币数量等信息。
"""
service = get_exchange_service(request)
cached = service.get_cached_rates()
return ServiceStatusResponse(
status="healthy" if cached else "degraded",
cache_valid=service.is_cache_valid(),
last_update=cached.cached_at.isoformat() if cached else None,
currencies_count=len(cached.rates) if cached else 0,
api_keys_count=len(service.api_keys),
update_interval_seconds=service.settings.CACHE_UPDATE_INTERVAL
)
# ==================== 强制刷新缓存 ====================
@router.post("/refresh")
async def refresh_cache(request: Request):
"""
强制刷新汇率缓存
立即从 API 获取最新汇率数据,不等待定时任务。
注意:频繁调用可能导致 API 限流。
"""
service = get_exchange_service(request)
logger.info("收到手动刷新缓存请求")
success = await service.update_cache()
if success:
cached = service.get_cached_rates()
return {
"success": True,
"message": "缓存刷新成功",
"currencies_count": len(cached.rates) if cached else 0,
"cached_at": cached.cached_at.isoformat() if cached else None
}
else:
raise HTTPException(
status_code=503,
detail="刷新缓存失败,请稍后重试。"
)
|