Initial commit: Daily News App
企業內部新聞彙整與分析系統 - 自動新聞抓取 (Digitimes, 經濟日報, 工商時報) - AI 智慧摘要 (OpenAI/Claude/Ollama) - 群組管理與訂閱通知 - 已清理 Python 快取檔案 🤖 Generated with [Claude Code](https://claude.com/claude-code) Co-Authored-By: Claude <noreply@anthropic.com>
This commit is contained in:
70
app/schemas/group.py
Normal file
70
app/schemas/group.py
Normal file
@@ -0,0 +1,70 @@
|
||||
"""
|
||||
群組與關鍵字 Pydantic Schema
|
||||
"""
|
||||
from datetime import datetime
|
||||
from typing import Optional, Literal
|
||||
from pydantic import BaseModel, Field
|
||||
|
||||
from app.schemas.user import PaginationResponse
|
||||
|
||||
|
||||
# ===== Keyword =====
|
||||
class KeywordBase(BaseModel):
|
||||
keyword: str = Field(..., max_length=100)
|
||||
|
||||
|
||||
class KeywordCreate(KeywordBase):
|
||||
pass
|
||||
|
||||
|
||||
class KeywordResponse(KeywordBase):
|
||||
id: int
|
||||
is_active: bool
|
||||
|
||||
class Config:
|
||||
from_attributes = True
|
||||
|
||||
|
||||
# ===== Group =====
|
||||
class GroupBase(BaseModel):
|
||||
name: str = Field(..., max_length=100)
|
||||
description: Optional[str] = None
|
||||
category: Literal["industry", "topic"]
|
||||
|
||||
|
||||
class GroupCreate(GroupBase):
|
||||
ai_background: Optional[str] = None
|
||||
ai_prompt: Optional[str] = None
|
||||
keywords: Optional[list[str]] = None
|
||||
|
||||
|
||||
class GroupUpdate(BaseModel):
|
||||
name: Optional[str] = Field(None, max_length=100)
|
||||
description: Optional[str] = None
|
||||
category: Optional[Literal["industry", "topic"]] = None
|
||||
ai_background: Optional[str] = None
|
||||
ai_prompt: Optional[str] = None
|
||||
is_active: Optional[bool] = None
|
||||
|
||||
|
||||
class GroupResponse(GroupBase):
|
||||
id: int
|
||||
is_active: bool
|
||||
keyword_count: Optional[int] = 0
|
||||
subscriber_count: Optional[int] = 0
|
||||
|
||||
class Config:
|
||||
from_attributes = True
|
||||
|
||||
|
||||
class GroupDetailResponse(GroupResponse):
|
||||
ai_background: Optional[str] = None
|
||||
ai_prompt: Optional[str] = None
|
||||
keywords: list[KeywordResponse] = []
|
||||
created_at: datetime
|
||||
updated_at: datetime
|
||||
|
||||
|
||||
class GroupListResponse(BaseModel):
|
||||
data: list[GroupResponse]
|
||||
pagination: PaginationResponse
|
||||
126
app/schemas/report.py
Normal file
126
app/schemas/report.py
Normal file
@@ -0,0 +1,126 @@
|
||||
"""
|
||||
報告相關 Pydantic Schema
|
||||
"""
|
||||
from datetime import datetime, date
|
||||
from typing import Optional, Literal
|
||||
from pydantic import BaseModel, Field
|
||||
|
||||
from app.schemas.user import PaginationResponse
|
||||
|
||||
|
||||
# ===== Article (簡化版) =====
|
||||
class ArticleBrief(BaseModel):
|
||||
id: int
|
||||
title: str
|
||||
source_name: str
|
||||
url: str
|
||||
published_at: Optional[datetime] = None
|
||||
|
||||
class Config:
|
||||
from_attributes = True
|
||||
|
||||
|
||||
class ArticleInReport(ArticleBrief):
|
||||
is_included: bool = True
|
||||
|
||||
|
||||
# ===== Report =====
|
||||
class ReportBase(BaseModel):
|
||||
title: str = Field(..., max_length=200)
|
||||
|
||||
|
||||
class ReportUpdate(BaseModel):
|
||||
title: Optional[str] = Field(None, max_length=200)
|
||||
edited_summary: Optional[str] = None
|
||||
article_selections: Optional[list[dict]] = None # [{article_id: int, is_included: bool}]
|
||||
|
||||
|
||||
class GroupBrief(BaseModel):
|
||||
id: int
|
||||
name: str
|
||||
category: str
|
||||
|
||||
class Config:
|
||||
from_attributes = True
|
||||
|
||||
|
||||
class ReportResponse(ReportBase):
|
||||
id: int
|
||||
report_date: date
|
||||
status: Literal["draft", "pending", "published", "delayed"]
|
||||
group: GroupBrief
|
||||
article_count: Optional[int] = 0
|
||||
published_at: Optional[datetime] = None
|
||||
|
||||
class Config:
|
||||
from_attributes = True
|
||||
|
||||
|
||||
class ReportDetailResponse(ReportResponse):
|
||||
ai_summary: Optional[str] = None
|
||||
edited_summary: Optional[str] = None
|
||||
articles: list[ArticleInReport] = []
|
||||
is_favorited: Optional[bool] = False
|
||||
comment_count: Optional[int] = 0
|
||||
created_at: datetime
|
||||
updated_at: datetime
|
||||
|
||||
|
||||
class ReportReviewResponse(ReportResponse):
|
||||
"""專員審核用"""
|
||||
ai_summary: Optional[str] = None
|
||||
edited_summary: Optional[str] = None
|
||||
articles: list[ArticleInReport] = []
|
||||
|
||||
|
||||
class ReportListResponse(BaseModel):
|
||||
data: list[ReportResponse]
|
||||
pagination: PaginationResponse
|
||||
|
||||
|
||||
class PublishResponse(BaseModel):
|
||||
published_at: datetime
|
||||
notifications_sent: int
|
||||
|
||||
|
||||
class RegenerateSummaryResponse(BaseModel):
|
||||
ai_summary: str
|
||||
|
||||
|
||||
# ===== Article Full =====
|
||||
class ArticleSourceBrief(BaseModel):
|
||||
id: int
|
||||
name: str
|
||||
|
||||
class Config:
|
||||
from_attributes = True
|
||||
|
||||
|
||||
class ArticleResponse(BaseModel):
|
||||
id: int
|
||||
title: str
|
||||
source: ArticleSourceBrief
|
||||
url: str
|
||||
published_at: Optional[datetime] = None
|
||||
crawled_at: datetime
|
||||
|
||||
class Config:
|
||||
from_attributes = True
|
||||
|
||||
|
||||
class MatchedGroup(BaseModel):
|
||||
group_id: int
|
||||
group_name: str
|
||||
matched_keywords: list[str]
|
||||
|
||||
|
||||
class ArticleDetailResponse(ArticleResponse):
|
||||
content: Optional[str] = None
|
||||
summary: Optional[str] = None
|
||||
author: Optional[str] = None
|
||||
matched_groups: list[MatchedGroup] = []
|
||||
|
||||
|
||||
class ArticleListResponse(BaseModel):
|
||||
data: list[ArticleResponse]
|
||||
pagination: PaginationResponse
|
||||
88
app/schemas/user.py
Normal file
88
app/schemas/user.py
Normal file
@@ -0,0 +1,88 @@
|
||||
"""
|
||||
用戶相關 Pydantic Schema
|
||||
"""
|
||||
from datetime import datetime
|
||||
from typing import Optional, Literal
|
||||
from pydantic import BaseModel, EmailStr, Field
|
||||
|
||||
|
||||
# ===== Pagination =====
|
||||
class PaginationResponse(BaseModel):
|
||||
page: int
|
||||
limit: int
|
||||
total: int
|
||||
total_pages: int
|
||||
|
||||
|
||||
# ===== Role =====
|
||||
class RoleBase(BaseModel):
|
||||
code: str
|
||||
name: str
|
||||
description: Optional[str] = None
|
||||
|
||||
|
||||
class RoleResponse(RoleBase):
|
||||
id: int
|
||||
|
||||
class Config:
|
||||
from_attributes = True
|
||||
|
||||
|
||||
# ===== User =====
|
||||
class UserBase(BaseModel):
|
||||
username: str = Field(..., min_length=2, max_length=50)
|
||||
display_name: str = Field(..., min_length=1, max_length=100)
|
||||
email: Optional[EmailStr] = None
|
||||
|
||||
|
||||
class UserCreate(UserBase):
|
||||
password: Optional[str] = Field(None, min_length=6, description="本地帳號必填")
|
||||
auth_type: Literal["ad", "local"] = "local"
|
||||
role_id: int
|
||||
|
||||
|
||||
class UserUpdate(BaseModel):
|
||||
display_name: Optional[str] = Field(None, max_length=100)
|
||||
email: Optional[EmailStr] = None
|
||||
role_id: Optional[int] = None
|
||||
is_active: Optional[bool] = None
|
||||
password: Optional[str] = Field(None, min_length=6, description="僅本地帳號可修改")
|
||||
|
||||
|
||||
class UserResponse(UserBase):
|
||||
id: int
|
||||
auth_type: str
|
||||
role: RoleResponse
|
||||
is_active: bool
|
||||
last_login_at: Optional[datetime] = None
|
||||
created_at: datetime
|
||||
|
||||
class Config:
|
||||
from_attributes = True
|
||||
|
||||
|
||||
class UserListResponse(BaseModel):
|
||||
data: list[UserResponse]
|
||||
pagination: "PaginationResponse"
|
||||
|
||||
|
||||
# ===== Auth =====
|
||||
class LoginRequest(BaseModel):
|
||||
username: str
|
||||
password: str
|
||||
auth_type: Literal["ad", "local"] = "ad"
|
||||
|
||||
|
||||
class LoginResponse(BaseModel):
|
||||
token: str
|
||||
user: UserResponse
|
||||
|
||||
|
||||
class TokenPayload(BaseModel):
|
||||
user_id: int
|
||||
username: str
|
||||
role: str
|
||||
exp: datetime
|
||||
|
||||
|
||||
|
||||
Reference in New Issue
Block a user