- PostLike domain entity (post_id, liked_by) with BaseEntity integration
- Post entity: add like_count field (default 0) and to_dict serialization
- PostRepository interface: add get_like, add_like, remove_like methods
- TogglePostLikeUseCase: toggle logic (like → unlike, unlike → like)
- PostResponseDTO/PostResponseSchema: add like_count field
- PostLikeORM model with FK to posts and cascade delete
- SQLAlchemyPostRepository: implement like query/add/remove with ORM mapping
- DI provider registration for TogglePostLikeUseCase
- API endpoint POST /api/v1/posts/{id}/like (auth required)
- Unit tests: PostLike entity, Post.like_count, TogglePostLikeUseCase (7 tests)
- API tests: POST /api/v1/posts/{id}/like (4 tests)
- Test model files: FEATURE_LIKES.md, TEST_MODEL.md updated
126 lines
2.8 KiB
Python
126 lines
2.8 KiB
Python
"""API schemas for posts.
|
|
|
|
This module defines Pydantic schemas for request/response validation
|
|
in the posts API endpoints.
|
|
"""
|
|
|
|
from datetime import datetime
|
|
from uuid import UUID
|
|
|
|
from pydantic import BaseModel, ConfigDict, Field
|
|
|
|
|
|
class PostBaseSchema(BaseModel):
|
|
"""Base schema for posts.
|
|
|
|
Contains common fields shared across post schemas.
|
|
|
|
Attributes:
|
|
title: Post title (3-200 characters).
|
|
content: Post content (10-50000 characters).
|
|
"""
|
|
|
|
model_config = ConfigDict(from_attributes=True)
|
|
|
|
title: str = Field(..., min_length=3, max_length=200)
|
|
content: str = Field(..., min_length=10, max_length=50000)
|
|
|
|
|
|
class PostCreateSchema(PostBaseSchema):
|
|
"""Schema for creating a post.
|
|
|
|
Extends base schema with creation-specific fields.
|
|
|
|
Attributes:
|
|
tags: List of tags for categorization.
|
|
"""
|
|
|
|
tags: list[str] = Field(default_factory=list)
|
|
|
|
|
|
class PostUpdateSchema(BaseModel):
|
|
"""Schema for updating a post.
|
|
|
|
All fields are optional for partial updates.
|
|
|
|
Attributes:
|
|
title: Optional new title.
|
|
content: Optional new content.
|
|
tags: Optional new tags list.
|
|
"""
|
|
|
|
model_config = ConfigDict(from_attributes=True)
|
|
|
|
title: str | None = Field(None, min_length=3, max_length=200)
|
|
content: str | None = Field(None, min_length=10, max_length=50000)
|
|
tags: list[str] | None = None
|
|
|
|
|
|
class PostResponseSchema(BaseModel):
|
|
"""Schema for post response.
|
|
|
|
Complete post data for API responses.
|
|
|
|
Attributes:
|
|
id: Unique post identifier.
|
|
title: Post title.
|
|
content: Post content.
|
|
slug: URL-friendly slug.
|
|
author_id: Author identifier.
|
|
published: Publication status.
|
|
tags: List of tags.
|
|
created_at: Creation timestamp.
|
|
updated_at: Last update timestamp.
|
|
"""
|
|
|
|
model_config = ConfigDict(from_attributes=True)
|
|
|
|
id: UUID
|
|
title: str
|
|
content: str
|
|
slug: str
|
|
author_id: str
|
|
published: bool
|
|
like_count: int = 0
|
|
tags: list[str]
|
|
created_at: datetime
|
|
updated_at: datetime
|
|
|
|
|
|
class PostListResponseSchema(BaseModel):
|
|
"""Schema for list of posts response.
|
|
|
|
Paginated response for list endpoints.
|
|
|
|
Attributes:
|
|
items: List of post items.
|
|
total: Total number of items.
|
|
"""
|
|
|
|
items: list[PostResponseSchema]
|
|
total: int
|
|
|
|
|
|
class PostSearchSchema(BaseModel):
|
|
"""Schema for searching posts.
|
|
|
|
Search query parameters.
|
|
|
|
Attributes:
|
|
query: Search query string (1-100 characters).
|
|
"""
|
|
|
|
query: str = Field(..., min_length=1, max_length=100)
|
|
|
|
|
|
class PostPublishSchema(BaseModel):
|
|
"""Schema for publishing/unpublishing a post.
|
|
|
|
Publication status toggle.
|
|
|
|
Attributes:
|
|
published: Desired publication status.
|
|
"""
|
|
|
|
published: bool
|