document.py 2.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172
  1. from datetime import datetime
  2. from typing import Literal, Optional
  3. from pydantic import BaseModel, Field, model_validator
  4. # ---------- 局部块更新 ----------
  5. class BlockUpdate(BaseModel):
  6. level: int = Field(..., ge=1, le=6, description="标题等级 1=H1 … 6=H6")
  7. index: int = Field(..., ge=0, description="该等级标题在文档中的顺序下标(从 0 开始)")
  8. content: str = Field(..., description="该块的新内容(含标题行本身)")
  9. # ---------- 请求 ----------
  10. class CreateDocumentRequest(BaseModel):
  11. title: str = Field(..., max_length=255)
  12. content: str = Field(..., description="文档内容(Markdown),最大 200KB")
  13. format: Literal["markdown"] = "markdown"
  14. session_id: Optional[str] = Field(None, alias="sessionId")
  15. template_id: Optional[str] = Field(None, alias="templateId")
  16. model_config = {"populate_by_name": True}
  17. class UpdateDocumentRequest(BaseModel):
  18. title: Optional[str] = Field(None, max_length=255)
  19. content: Optional[str] = Field(None, description="全量更新内容,与 blocks 二选一")
  20. blocks: Optional[list[BlockUpdate]] = Field(None, description="局部块更新,与 content 二选一")
  21. @model_validator(mode="after")
  22. def check_content_or_blocks(self) -> "UpdateDocumentRequest":
  23. if self.content is not None and self.blocks is not None:
  24. raise ValueError("content 与 blocks 不能同时传入")
  25. return self
  26. # ---------- 响应 ----------
  27. class DocumentResponse(BaseModel):
  28. id: str
  29. title: str
  30. content: str
  31. format: str
  32. session_id: Optional[str] = Field(None, serialization_alias="sessionId")
  33. template_id: Optional[str] = Field(None, serialization_alias="templateId")
  34. created_by: Optional[str] = Field(None, serialization_alias="createdBy")
  35. created_at: datetime = Field(..., serialization_alias="createdAt")
  36. updated_at: datetime = Field(..., serialization_alias="updatedAt")
  37. model_config = {"from_attributes": True, "populate_by_name": True}
  38. class DocumentListItem(BaseModel):
  39. id: str
  40. title: str
  41. template_id: Optional[str] = Field(None, serialization_alias="templateId")
  42. created_at: datetime = Field(..., serialization_alias="createdAt")
  43. updated_at: datetime = Field(..., serialization_alias="updatedAt")
  44. model_config = {"from_attributes": True, "populate_by_name": True}
  45. class Pagination(BaseModel):
  46. page: int
  47. page_size: int = Field(..., serialization_alias="pageSize")
  48. total: int
  49. total_pages: int = Field(..., serialization_alias="totalPages")
  50. model_config = {"populate_by_name": True}