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:
donald
2025-12-03 23:53:24 +08:00
commit db0f0bbfe7
50 changed files with 11883 additions and 0 deletions

70
app/schemas/group.py Normal file
View 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
View 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
View 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