234 lines
8.0 KiB
Python
234 lines
8.0 KiB
Python
from __future__ import annotations
|
||
|
||
from contextlib import asynccontextmanager
|
||
from pathlib import Path
|
||
|
||
from app.config import settings
|
||
from app.dependencies.database import engine, get_redis, redis_client
|
||
from app.dependencies.fetcher import get_fetcher
|
||
from app.dependencies.scheduler import start_scheduler, stop_scheduler
|
||
from app.log import logger
|
||
from app.router import (
|
||
api_v1_router,
|
||
api_v2_router,
|
||
auth_router,
|
||
chat_router,
|
||
fetcher_router,
|
||
file_router,
|
||
lio_router,
|
||
private_router,
|
||
redirect_api_router,
|
||
)
|
||
from app.router.redirect import redirect_router
|
||
from app.scheduler.cache_scheduler import start_cache_scheduler, stop_cache_scheduler
|
||
from app.scheduler.database_cleanup_scheduler import (
|
||
start_database_cleanup_scheduler,
|
||
stop_database_cleanup_scheduler,
|
||
)
|
||
from app.service.beatmap_download_service import download_service
|
||
from app.service.calculate_all_user_rank import calculate_user_rank
|
||
from app.service.create_banchobot import create_banchobot
|
||
from app.service.daily_challenge import daily_challenge_job, process_daily_challenge_top
|
||
from app.service.email_queue import start_email_processor, stop_email_processor
|
||
from app.service.geoip_scheduler import schedule_geoip_updates
|
||
from app.service.init_geoip import init_geoip
|
||
from app.service.load_achievements import load_achievements
|
||
from app.service.osu_rx_statistics import create_rx_statistics
|
||
from app.service.redis_message_system import redis_message_system
|
||
from app.utils import bg_tasks, utcnow
|
||
|
||
from fastapi import FastAPI, HTTPException, Request
|
||
from fastapi.exceptions import RequestValidationError
|
||
from fastapi.middleware.cors import CORSMiddleware
|
||
from fastapi.responses import JSONResponse
|
||
from fastapi_limiter import FastAPILimiter
|
||
import sentry_sdk
|
||
|
||
|
||
@asynccontextmanager
|
||
async def lifespan(app: FastAPI):
|
||
# on startup
|
||
await FastAPILimiter.init(get_redis())
|
||
await get_fetcher() # 初始化 fetcher
|
||
await init_geoip() # 初始化 GeoIP 数据库
|
||
await create_rx_statistics()
|
||
await calculate_user_rank(True)
|
||
start_scheduler()
|
||
schedule_geoip_updates() # 调度 GeoIP 定时更新任务
|
||
await daily_challenge_job()
|
||
await process_daily_challenge_top()
|
||
await create_banchobot()
|
||
await start_email_processor() # 启动邮件队列处理器
|
||
await download_service.start_health_check() # 启动下载服务健康检查
|
||
await start_cache_scheduler() # 启动缓存调度器
|
||
await start_database_cleanup_scheduler() # 启动数据库清理调度器
|
||
redis_message_system.start() # 启动 Redis 消息系统
|
||
load_achievements()
|
||
|
||
# 显示资源代理状态
|
||
if settings.enable_asset_proxy:
|
||
logger.info(f"Asset Proxy enabled - Domain: {settings.custom_asset_domain}")
|
||
|
||
# on shutdown
|
||
yield
|
||
bg_tasks.stop()
|
||
stop_scheduler()
|
||
redis_message_system.stop() # 停止 Redis 消息系统
|
||
await stop_cache_scheduler() # 停止缓存调度器
|
||
await stop_database_cleanup_scheduler() # 停止数据库清理调度器
|
||
await download_service.stop_health_check() # 停止下载服务健康检查
|
||
await stop_email_processor() # 停止邮件队列处理器
|
||
await engine.dispose()
|
||
await redis_client.aclose()
|
||
|
||
|
||
desc = f"""osu! API 模拟服务器,支持 osu! API v1, v2 和 osu!lazer 的绝大部分功能。
|
||
|
||
## 端点说明
|
||
|
||
所有 v2 API 均以 `/api/v2/` 开头,所有 v1 API 均以 `/api/v1` 开头(直接访问 `/api` 的 v1 API 会进行重定向)。
|
||
|
||
所有 g0v0-server 提供的额外 API(g0v0-api) 均以 `/api/private/` 开头。
|
||
|
||
## 鉴权
|
||
|
||
v2 API 采用 OAuth 2.0 鉴权,支持以下鉴权方式:
|
||
|
||
- `password` 密码鉴权,仅适用于 osu!lazer 客户端和前端等服务,需要提供用户的用户名和密码进行登录。
|
||
- `authorization_code` 授权码鉴权,适用于第三方应用,需要提供用户的授权码进行登录。
|
||
- `client_credentials` 客户端凭证鉴权,适用于服务端应用,需要提供客户端 ID 和客户端密钥进行登录。
|
||
|
||
使用 `password` 鉴权的具有全部权限。`authorization_code` 具有指定 scope 的权限。`client_credentials` 只有 `public` 权限。各接口需要的权限请查看每个 Endpoint 的 Authorization。
|
||
|
||
v1 API 采用 API Key 鉴权,将 API Key 放入 Query `k` 中。
|
||
|
||
{
|
||
'''
|
||
## 速率限制
|
||
|
||
所有 API 请求均受到速率限制,具体限制规则如下:
|
||
|
||
- 每分钟最多可以发送 1200 个请求
|
||
- 突发请求限制为每秒最多 200 个请求
|
||
'''
|
||
if settings.enable_rate_limit
|
||
else ""
|
||
}
|
||
|
||
## 参考
|
||
|
||
- v2 API 文档:[osu-web 文档](https://osu.ppy.sh/docs/index.html)
|
||
- v1 API 文档:[osu-api](https://github.com/ppy/osu-api/wiki)
|
||
""" # noqa: E501
|
||
|
||
# 检查 New Relic 配置文件是否存在,如果存在则初始化 New Relic
|
||
newrelic_config_path = Path("newrelic.ini")
|
||
if newrelic_config_path.exists():
|
||
try:
|
||
import newrelic.agent
|
||
|
||
environment = settings.new_relic_environment or ("production" if not settings.debug else "development")
|
||
|
||
newrelic.agent.initialize(newrelic_config_path, environment)
|
||
logger.info(f"[NewRelic] Enabled, environment: {environment}")
|
||
except ImportError:
|
||
logger.warning("[NewRelic] Config file found but 'newrelic' package is not installed")
|
||
except Exception as e:
|
||
logger.error(f"[NewRelic] Initialization failed: {e}")
|
||
else:
|
||
logger.info("[NewRelic] No newrelic.ini config file found, skipping initialization")
|
||
|
||
if settings.sentry_dsn is not None:
|
||
sentry_sdk.init(
|
||
dsn=str(settings.sentry_dsn),
|
||
send_default_pii=False,
|
||
environment="production" if not settings.debug else "development",
|
||
)
|
||
|
||
app = FastAPI(
|
||
title="g0v0-server",
|
||
version="0.1.0",
|
||
lifespan=lifespan,
|
||
description=desc,
|
||
)
|
||
|
||
|
||
app.include_router(api_v2_router)
|
||
app.include_router(api_v1_router)
|
||
app.include_router(chat_router)
|
||
app.include_router(redirect_api_router)
|
||
app.include_router(fetcher_router)
|
||
app.include_router(file_router)
|
||
app.include_router(auth_router)
|
||
app.include_router(private_router)
|
||
app.include_router(lio_router)
|
||
|
||
# from app.signalr import signalr_router
|
||
# app.include_router(signalr_router)
|
||
|
||
# CORS 配置
|
||
origins = []
|
||
for url in [*settings.cors_urls, settings.server_url]:
|
||
origins.append(str(url))
|
||
origins.append(str(url).removesuffix("/"))
|
||
if settings.frontend_url:
|
||
origins.append(str(settings.frontend_url))
|
||
origins.append(str(settings.frontend_url).removesuffix("/"))
|
||
app.add_middleware(
|
||
CORSMiddleware,
|
||
allow_origins=origins,
|
||
allow_credentials=True,
|
||
allow_methods=["*"],
|
||
allow_headers=["*"],
|
||
)
|
||
|
||
if settings.frontend_url is not None:
|
||
app.include_router(redirect_router)
|
||
|
||
|
||
@app.get("/", include_in_schema=False)
|
||
async def root():
|
||
"""根端点"""
|
||
return {"message": "osu! API 模拟服务器正在运行"}
|
||
|
||
|
||
@app.get("/health", include_in_schema=False)
|
||
async def health_check():
|
||
"""健康检查端点"""
|
||
return {"status": "ok", "timestamp": utcnow().isoformat()}
|
||
|
||
|
||
@app.exception_handler(RequestValidationError)
|
||
async def validation_exception_handler(request: Request, exc: RequestValidationError):
|
||
return JSONResponse(
|
||
status_code=422,
|
||
content={
|
||
"error": exc.errors(),
|
||
},
|
||
)
|
||
|
||
|
||
@app.exception_handler(HTTPException)
|
||
async def http_exception_handler(requst: Request, exc: HTTPException):
|
||
return JSONResponse(status_code=exc.status_code, content={"error": exc.detail})
|
||
|
||
|
||
if settings.secret_key == "your_jwt_secret_here":
|
||
logger.warning("jwt_secret_key is unset. Your server is unsafe. Use this command to generate: openssl rand -hex 32")
|
||
if settings.osu_web_client_secret == "your_osu_web_client_secret_here":
|
||
logger.warning(
|
||
"osu_web_client_secret is unset. Your server is unsafe. Use this command to generate: openssl rand -hex 40"
|
||
)
|
||
|
||
if __name__ == "__main__":
|
||
import uvicorn
|
||
|
||
uvicorn.run(
|
||
"main:app",
|
||
host=settings.host,
|
||
port=settings.port,
|
||
reload=settings.debug,
|
||
log_config=None, # 禁用uvicorn默认日志配置
|
||
access_log=True, # 启用访问日志
|
||
)
|