- backend: FastAPI + SQLAlchemy 2.0(async) + asyncpg + Alembic - 7 API routes: auth/me/articles/sources/bookmarks/subscriptions/admin - models: User/Source/Article/Bookmark/Subscription/ApiToken - services: RSS fetcher (feedparser) + Tencent TMT translator with quota + cache + local NLLB fallback - workers: APScheduler + asyncio pipeline (fetch -> dedupe -> insert -> translate) - seed scripts: create_user, seed_sources (5 RSS: Reuters/BBC/Al Jazeera/NHK/DW) - frontend: Vue 3 + Vite + Naive UI + Pinia + vue-router - pages: Login, Feed (24h), ArticleDetail, Sources, Bookmarks, AdminSources - deploy: docker-compose (postgres/redis/api/worker/frontend/caddy) - docs: README, DEPLOY, architecture, acceptance
26 lines
826 B
Python
26 lines
826 B
Python
"""/sources 源列表(只读,所有登录用户可看)。"""
|
|
from __future__ import annotations
|
|
|
|
from fastapi import APIRouter, Depends
|
|
from sqlalchemy import select
|
|
from sqlalchemy.ext.asyncio import AsyncSession
|
|
|
|
from app.core.deps import get_current_user
|
|
from app.database import get_session
|
|
from app.models.source import Source
|
|
from app.models.user import User
|
|
from app.schemas.source import SourceOut
|
|
|
|
router = APIRouter(prefix="/sources", tags=["sources"])
|
|
|
|
|
|
@router.get("", response_model=list[SourceOut])
|
|
async def list_sources(
|
|
user: User = Depends(get_current_user), # noqa: ARG001
|
|
session: AsyncSession = Depends(get_session),
|
|
):
|
|
rows = (
|
|
await session.execute(select(Source).order_by(Source.priority.desc(), Source.name))
|
|
).scalars()
|
|
return [SourceOut.model_validate(s) for s in rows]
|