- backend(financial): 新增 /china/{ts_code}/snapshot API,返回昨日交易日的收盘价/市值/PE/PB/股息率等
- backend(schemas): 新增 TodaySnapshotResponse
- backend(main): 注册 orgs 路由 /api/v1/orgs
- backend(providers:finnhub): 归一化财报字段并计算 gross_margin/net_margin/ROA/ROE
- backend(providers:tushare): 股东户数报告期与财报期对齐
- backend(routers/financial): years 默认改为 10(最大 10)
- config: analysis-config.json 切换到 qwen-flash-2025-07-28
- frontend(report/[symbol]): 新增“昨日快照”卡片、限制展示期数为10、优化增长与阈值高亮、修正类名与标题处理
- frontend(reports/[id]): 统一 period 变量与计算,修正表格 key
- frontend(hooks): 新增 useChinaSnapshot 钩子与类型
- scripts: dev.sh 增加调试输出
59 lines
1.9 KiB
Python
59 lines
1.9 KiB
Python
"""
|
|
FastAPI application entrypoint
|
|
"""
|
|
import logging
|
|
from fastapi import FastAPI
|
|
from fastapi.middleware.cors import CORSMiddleware
|
|
|
|
from app.core.config import settings
|
|
from app.routers.config import router as config_router
|
|
from app.routers.financial import router as financial_router
|
|
from app.routers.orgs import router as orgs_router
|
|
|
|
# Configure logging to ensure our app logs show up in development
|
|
import sys
|
|
|
|
# Force our logging configuration to override uvicorn's
|
|
class ForcefulHandler(logging.Handler):
|
|
def emit(self, record):
|
|
# Force output to stdout regardless of uvicorn's configuration
|
|
print(f"[APP] {record.getMessage()}", file=sys.stdout, flush=True)
|
|
|
|
# Set up our forceful handler for data providers
|
|
forceful_handler = ForcefulHandler()
|
|
forceful_handler.setLevel(logging.DEBUG)
|
|
|
|
# Configure data providers logger with forceful output
|
|
data_providers_logger = logging.getLogger('app.data_providers')
|
|
data_providers_logger.setLevel(logging.DEBUG)
|
|
data_providers_logger.addHandler(forceful_handler)
|
|
|
|
# Also set up for the main app logger
|
|
app_logger = logging.getLogger('app')
|
|
app_logger.setLevel(logging.INFO)
|
|
app_logger.addHandler(forceful_handler)
|
|
|
|
# Ensure our handlers are not suppressed
|
|
data_providers_logger.propagate = False
|
|
app_logger.propagate = False
|
|
|
|
app = FastAPI(title=settings.APP_NAME, version=settings.APP_VERSION)
|
|
|
|
# CORS
|
|
app.add_middleware(
|
|
CORSMiddleware,
|
|
allow_origins=["*"],
|
|
allow_credentials=True,
|
|
allow_methods=["*"],
|
|
allow_headers=["*"],
|
|
)
|
|
|
|
# Routers
|
|
app.include_router(config_router, prefix=f"{settings.API_V1_STR}/config", tags=["config"])
|
|
app.include_router(financial_router, prefix=f"{settings.API_V1_STR}/financials", tags=["financials"])
|
|
app.include_router(orgs_router, prefix=f"{settings.API_V1_STR}/orgs", tags=["orgs"])
|
|
|
|
@app.get("/")
|
|
async def root():
|
|
return {"status": "ok", "name": settings.APP_NAME, "version": settings.APP_VERSION}
|