feat(fetcher): add data fetcher
This commit is contained in:
@@ -36,5 +36,13 @@ class Settings:
|
|||||||
SIGNALR_NEGOTIATE_TIMEOUT: int = int(os.getenv("SIGNALR_NEGOTIATE_TIMEOUT", "30"))
|
SIGNALR_NEGOTIATE_TIMEOUT: int = int(os.getenv("SIGNALR_NEGOTIATE_TIMEOUT", "30"))
|
||||||
SIGNALR_PING_INTERVAL: int = int(os.getenv("SIGNALR_PING_INTERVAL", "120"))
|
SIGNALR_PING_INTERVAL: int = int(os.getenv("SIGNALR_PING_INTERVAL", "120"))
|
||||||
|
|
||||||
|
# Fetcher 设置
|
||||||
|
FETCHER_CLIENT_ID: str = os.getenv("FETCHER_CLIENT_ID", "")
|
||||||
|
FETCHER_CLIENT_SECRET: str = os.getenv("FETCHER_CLIENT_SECRET", "")
|
||||||
|
FETCHER_SCOPES: list[str] = os.getenv("FETCHER_SCOPES", "public").split(",")
|
||||||
|
FETCHER_CALLBACK_URL: str = os.getenv(
|
||||||
|
"FETCHER_CALLBACK_URL", "http://localhost:8000/fetcher/callback"
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
settings = Settings()
|
settings = Settings()
|
||||||
|
|||||||
29
app/dependencies/fetcher.py
Normal file
29
app/dependencies/fetcher.py
Normal file
@@ -0,0 +1,29 @@
|
|||||||
|
from __future__ import annotations
|
||||||
|
|
||||||
|
from app.config import settings
|
||||||
|
from app.dependencies.database import get_redis
|
||||||
|
from app.fetcher import Fetcher
|
||||||
|
|
||||||
|
fetcher: Fetcher | None = None
|
||||||
|
|
||||||
|
|
||||||
|
def get_fetcher() -> Fetcher:
|
||||||
|
global fetcher
|
||||||
|
if fetcher is None:
|
||||||
|
fetcher = Fetcher(
|
||||||
|
settings.FETCHER_CLIENT_ID,
|
||||||
|
settings.FETCHER_CLIENT_SECRET,
|
||||||
|
settings.FETCHER_SCOPES,
|
||||||
|
settings.FETCHER_CALLBACK_URL,
|
||||||
|
)
|
||||||
|
redis = get_redis()
|
||||||
|
if redis:
|
||||||
|
access_token = redis.get(f"fetcher:access_token:{fetcher.client_id}")
|
||||||
|
if access_token:
|
||||||
|
fetcher.access_token = str(access_token)
|
||||||
|
refresh_token = redis.get(f"fetcher:refresh_token:{fetcher.client_id}")
|
||||||
|
if refresh_token:
|
||||||
|
fetcher.refresh_token = str(refresh_token)
|
||||||
|
if not fetcher.access_token or not fetcher.refresh_token:
|
||||||
|
print("Login to initialize fetcher:", fetcher.authorize_url)
|
||||||
|
return fetcher
|
||||||
10
app/fetcher/__init__.py
Normal file
10
app/fetcher/__init__.py
Normal file
@@ -0,0 +1,10 @@
|
|||||||
|
from __future__ import annotations
|
||||||
|
|
||||||
|
from .beatmap import BeatmapFetcher
|
||||||
|
from .beatmapset import BeatmapsetFetcher
|
||||||
|
|
||||||
|
|
||||||
|
class Fetcher(BeatmapFetcher, BeatmapsetFetcher):
|
||||||
|
"""A class that combines all fetchers for easy access."""
|
||||||
|
|
||||||
|
pass
|
||||||
99
app/fetcher/_base.py
Normal file
99
app/fetcher/_base.py
Normal file
@@ -0,0 +1,99 @@
|
|||||||
|
from __future__ import annotations
|
||||||
|
|
||||||
|
import time
|
||||||
|
|
||||||
|
from app.dependencies.database import get_redis
|
||||||
|
|
||||||
|
from httpx import AsyncClient
|
||||||
|
|
||||||
|
|
||||||
|
class BaseFetcher:
|
||||||
|
def __init__(
|
||||||
|
self,
|
||||||
|
client_id: str,
|
||||||
|
client_secret: str,
|
||||||
|
scope: list[str] = ["public"],
|
||||||
|
callback_url: str = "",
|
||||||
|
):
|
||||||
|
self.client_id = client_id
|
||||||
|
self.client_secret = client_secret
|
||||||
|
self.access_token: str = ""
|
||||||
|
self.refresh_token: str = ""
|
||||||
|
self.token_expiry: int = 0
|
||||||
|
self.callback_url: str = callback_url
|
||||||
|
self.scope = scope
|
||||||
|
|
||||||
|
@property
|
||||||
|
def authorize_url(self) -> str:
|
||||||
|
return (
|
||||||
|
f"https://osu.ppy.sh/oauth/authorize?client_id={self.client_id}"
|
||||||
|
f"&response_type=code&scope={' '.join(self.scope)}"
|
||||||
|
f"&redirect_uri={self.callback_url}"
|
||||||
|
)
|
||||||
|
|
||||||
|
@property
|
||||||
|
def header(self) -> dict[str, str]:
|
||||||
|
return {
|
||||||
|
"Authorization": f"Bearer {self.access_token}",
|
||||||
|
"Content-Type": "application/json",
|
||||||
|
}
|
||||||
|
|
||||||
|
def is_token_expired(self) -> bool:
|
||||||
|
return self.token_expiry <= int(time.time())
|
||||||
|
|
||||||
|
async def grant_access_token(self, code: str) -> None:
|
||||||
|
async with AsyncClient() as client:
|
||||||
|
response = await client.post(
|
||||||
|
"https://osu.ppy.sh/oauth/token",
|
||||||
|
data={
|
||||||
|
"client_id": self.client_id,
|
||||||
|
"client_secret": self.client_secret,
|
||||||
|
"grant_type": "authorization_code",
|
||||||
|
"redirect_uri": self.callback_url,
|
||||||
|
"code": code,
|
||||||
|
},
|
||||||
|
)
|
||||||
|
response.raise_for_status()
|
||||||
|
token_data = response.json()
|
||||||
|
self.access_token = token_data["access_token"]
|
||||||
|
self.refresh_token = token_data.get("refresh_token", "")
|
||||||
|
self.token_expiry = int(time.time()) + token_data["expires_in"]
|
||||||
|
redis = get_redis()
|
||||||
|
if redis:
|
||||||
|
redis.set(
|
||||||
|
f"fetcher:access_token:{self.client_id}",
|
||||||
|
self.access_token,
|
||||||
|
ex=token_data["expires_in"],
|
||||||
|
)
|
||||||
|
redis.set(
|
||||||
|
f"fetcher:refresh_token:{self.client_id}",
|
||||||
|
self.refresh_token,
|
||||||
|
)
|
||||||
|
|
||||||
|
async def refresh_access_token(self) -> None:
|
||||||
|
async with AsyncClient() as client:
|
||||||
|
response = await client.post(
|
||||||
|
"https://osu.ppy.sh/oauth/token",
|
||||||
|
data={
|
||||||
|
"client_id": self.client_id,
|
||||||
|
"client_secret": self.client_secret,
|
||||||
|
"grant_type": "refresh_token",
|
||||||
|
"refresh_token": self.refresh_token,
|
||||||
|
},
|
||||||
|
)
|
||||||
|
response.raise_for_status()
|
||||||
|
token_data = response.json()
|
||||||
|
self.access_token = token_data["access_token"]
|
||||||
|
self.refresh_token = token_data.get("refresh_token", "")
|
||||||
|
self.token_expiry = int(time.time()) + token_data["expires_in"]
|
||||||
|
redis = get_redis()
|
||||||
|
if redis:
|
||||||
|
redis.set(
|
||||||
|
f"fetcher:access_token:{self.client_id}",
|
||||||
|
self.access_token,
|
||||||
|
ex=token_data["expires_in"],
|
||||||
|
)
|
||||||
|
redis.set(
|
||||||
|
f"fetcher:refresh_token:{self.client_id}",
|
||||||
|
self.refresh_token,
|
||||||
|
)
|
||||||
18
app/fetcher/beatmap.py
Normal file
18
app/fetcher/beatmap.py
Normal file
@@ -0,0 +1,18 @@
|
|||||||
|
from __future__ import annotations
|
||||||
|
|
||||||
|
from app.database.beatmap import BeatmapResp
|
||||||
|
|
||||||
|
from ._base import BaseFetcher
|
||||||
|
|
||||||
|
from httpx import AsyncClient
|
||||||
|
|
||||||
|
|
||||||
|
class BeatmapFetcher(BaseFetcher):
|
||||||
|
async def get_beatmap(self, beatmap_id: int) -> BeatmapResp:
|
||||||
|
async with AsyncClient() as client:
|
||||||
|
response = await client.get(
|
||||||
|
f"https://osu.ppy.sh/api/v2/beatmaps/{beatmap_id}",
|
||||||
|
headers=self.header,
|
||||||
|
)
|
||||||
|
response.raise_for_status()
|
||||||
|
return BeatmapResp.model_validate(response.json())
|
||||||
18
app/fetcher/beatmapset.py
Normal file
18
app/fetcher/beatmapset.py
Normal file
@@ -0,0 +1,18 @@
|
|||||||
|
from __future__ import annotations
|
||||||
|
|
||||||
|
from app.database.beatmapset import BeatmapsetResp
|
||||||
|
|
||||||
|
from ._base import BaseFetcher
|
||||||
|
|
||||||
|
from httpx import AsyncClient
|
||||||
|
|
||||||
|
|
||||||
|
class BeatmapsetFetcher(BaseFetcher):
|
||||||
|
async def get_beatmap_set(self, beatmap_set_id: int) -> BeatmapsetResp:
|
||||||
|
async with AsyncClient() as client:
|
||||||
|
response = await client.get(
|
||||||
|
f"https://osu.ppy.sh/api/v2/beatmapsets/{beatmap_set_id}",
|
||||||
|
headers=self.header,
|
||||||
|
)
|
||||||
|
response.raise_for_status()
|
||||||
|
return BeatmapsetResp.model_validate(response.json())
|
||||||
@@ -7,6 +7,7 @@ from . import ( # pyright: ignore[reportUnusedImport] # noqa: F401
|
|||||||
)
|
)
|
||||||
from .api_router import router as api_router
|
from .api_router import router as api_router
|
||||||
from .auth import router as auth_router
|
from .auth import router as auth_router
|
||||||
|
from .fetcher import fetcher_router as fetcher_router
|
||||||
from .signalr import signalr_router as signalr_router
|
from .signalr import signalr_router as signalr_router
|
||||||
|
|
||||||
__all__ = ["api_router", "auth_router", "signalr_router"]
|
__all__ = ["api_router", "auth_router", "fetcher_router", "signalr_router"]
|
||||||
|
|||||||
14
app/router/fetcher.py
Normal file
14
app/router/fetcher.py
Normal file
@@ -0,0 +1,14 @@
|
|||||||
|
from __future__ import annotations
|
||||||
|
|
||||||
|
from app.dependencies.fetcher import get_fetcher
|
||||||
|
from app.fetcher import Fetcher
|
||||||
|
|
||||||
|
from fastapi import APIRouter, Depends
|
||||||
|
|
||||||
|
fetcher_router = APIRouter()
|
||||||
|
|
||||||
|
|
||||||
|
@fetcher_router.get("/callback")
|
||||||
|
async def callback(code: str, fetcher: Fetcher = Depends(get_fetcher)):
|
||||||
|
await fetcher.grant_access_token(code)
|
||||||
|
return {"message": "Login successful"}
|
||||||
5
main.py
5
main.py
@@ -5,7 +5,8 @@ from datetime import datetime
|
|||||||
|
|
||||||
from app.config import settings
|
from app.config import settings
|
||||||
from app.dependencies.database import create_tables, engine
|
from app.dependencies.database import create_tables, engine
|
||||||
from app.router import api_router, auth_router, signalr_router
|
from app.dependencies.fetcher import get_fetcher
|
||||||
|
from app.router import api_router, auth_router, fetcher_router, signalr_router
|
||||||
|
|
||||||
from fastapi import FastAPI
|
from fastapi import FastAPI
|
||||||
|
|
||||||
@@ -17,6 +18,7 @@ from fastapi import FastAPI
|
|||||||
async def lifespan(app: FastAPI):
|
async def lifespan(app: FastAPI):
|
||||||
# on startup
|
# on startup
|
||||||
await create_tables()
|
await create_tables()
|
||||||
|
get_fetcher() # 初始化 fetcher
|
||||||
# on shutdown
|
# on shutdown
|
||||||
yield
|
yield
|
||||||
await engine.dispose()
|
await engine.dispose()
|
||||||
@@ -25,6 +27,7 @@ async def lifespan(app: FastAPI):
|
|||||||
app = FastAPI(title="osu! API 模拟服务器", version="1.0.0", lifespan=lifespan)
|
app = FastAPI(title="osu! API 模拟服务器", version="1.0.0", lifespan=lifespan)
|
||||||
app.include_router(api_router, prefix="/api/v2")
|
app.include_router(api_router, prefix="/api/v2")
|
||||||
app.include_router(signalr_router, prefix="/signalr")
|
app.include_router(signalr_router, prefix="/signalr")
|
||||||
|
app.include_router(fetcher_router, prefix="/fetcher")
|
||||||
app.include_router(auth_router)
|
app.include_router(auth_router)
|
||||||
|
|
||||||
|
|
||||||
|
|||||||
@@ -10,6 +10,7 @@ dependencies = [
|
|||||||
"bcrypt>=4.1.2",
|
"bcrypt>=4.1.2",
|
||||||
"cryptography>=41.0.7",
|
"cryptography>=41.0.7",
|
||||||
"fastapi>=0.104.1",
|
"fastapi>=0.104.1",
|
||||||
|
"httpx>=0.28.1",
|
||||||
"msgpack>=1.1.1",
|
"msgpack>=1.1.1",
|
||||||
"passlib[bcrypt]>=1.7.4",
|
"passlib[bcrypt]>=1.7.4",
|
||||||
"pydantic[email]>=2.5.0",
|
"pydantic[email]>=2.5.0",
|
||||||
|
|||||||
@@ -2,8 +2,8 @@ aiomysql==0.2.0
|
|||||||
alembic==1.16.4
|
alembic==1.16.4
|
||||||
annotated-types==0.7.0
|
annotated-types==0.7.0
|
||||||
anyio==4.9.0
|
anyio==4.9.0
|
||||||
async-timeout==5.0.1
|
|
||||||
bcrypt==4.3.0
|
bcrypt==4.3.0
|
||||||
|
certifi==2025.7.14
|
||||||
cffi==1.17.1
|
cffi==1.17.1
|
||||||
cfgv==3.4.0
|
cfgv==3.4.0
|
||||||
click==8.2.1
|
click==8.2.1
|
||||||
@@ -17,7 +17,9 @@ fastapi==0.116.1
|
|||||||
filelock==3.18.0
|
filelock==3.18.0
|
||||||
greenlet==3.2.3
|
greenlet==3.2.3
|
||||||
h11==0.16.0
|
h11==0.16.0
|
||||||
|
httpcore==1.0.9
|
||||||
httptools==0.6.4
|
httptools==0.6.4
|
||||||
|
httpx==0.28.1
|
||||||
identify==2.6.12
|
identify==2.6.12
|
||||||
idna==3.10
|
idna==3.10
|
||||||
mako==1.3.10
|
mako==1.3.10
|
||||||
@@ -48,7 +50,6 @@ starlette==0.47.2
|
|||||||
typing-extensions==4.14.1
|
typing-extensions==4.14.1
|
||||||
typing-inspection==0.4.1
|
typing-inspection==0.4.1
|
||||||
uvicorn==0.35.0
|
uvicorn==0.35.0
|
||||||
uvloop==0.21.0
|
|
||||||
virtualenv==20.32.0
|
virtualenv==20.32.0
|
||||||
watchfiles==1.1.0
|
watchfiles==1.1.0
|
||||||
websockets==15.0.1
|
websockets==15.0.1
|
||||||
|
|||||||
@@ -8,7 +8,7 @@ from __future__ import annotations
|
|||||||
import os
|
import os
|
||||||
|
|
||||||
from dotenv import load_dotenv
|
from dotenv import load_dotenv
|
||||||
import requests
|
import httpx as requests
|
||||||
|
|
||||||
# 加载 .env 文件
|
# 加载 .env 文件
|
||||||
load_dotenv()
|
load_dotenv()
|
||||||
@@ -173,7 +173,7 @@ def main():
|
|||||||
scores_data = get_beatmap_scores(token_data["access_token"], 1)
|
scores_data = get_beatmap_scores(token_data["access_token"], 1)
|
||||||
if scores_data:
|
if scores_data:
|
||||||
print(f"谱面成绩总数: {len(scores_data['scores'])}")
|
print(f"谱面成绩总数: {len(scores_data['scores'])}")
|
||||||
if scores_data['userScore']:
|
if scores_data["userScore"]:
|
||||||
print("用户在该谱面有成绩记录")
|
print("用户在该谱面有成绩记录")
|
||||||
print(f"用户成绩 ID: {scores_data['userScore']['id']}")
|
print(f"用户成绩 ID: {scores_data['userScore']['id']}")
|
||||||
print(f"用户成绩分数: {scores_data['userScore']['total_score']}")
|
print(f"用户成绩分数: {scores_data['userScore']['total_score']}")
|
||||||
@@ -196,4 +196,4 @@ def main():
|
|||||||
|
|
||||||
|
|
||||||
if __name__ == "__main__":
|
if __name__ == "__main__":
|
||||||
main()
|
main()
|
||||||
|
|||||||
Reference in New Issue
Block a user