Files
social-app/backend/src/v1/schedule_items/schemas.py
T

179 lines
5.1 KiB
Python
Raw Normal View History

2026-02-28 11:03:29 +08:00
from __future__ import annotations
from datetime import datetime
from typing import ClassVar
2026-02-28 11:03:29 +08:00
from uuid import UUID
from zoneinfo import ZoneInfo, ZoneInfoNotFoundError
2026-02-28 11:03:29 +08:00
from pydantic import BaseModel, ConfigDict, Field, field_validator
2026-02-28 11:03:29 +08:00
from schemas.inbox.messages import (
CalendarContent,
CalendarDeleteContent,
CalendarInviteContent,
CalendarUpdateContent,
parse_calendar_content,
)
from schemas.schedule.items import (
AttachmentType,
ScheduleItemMetadata,
ScheduleItemMetadataAttachment,
ScheduleItemSourceType,
ScheduleItemStatus,
)
__all__ = [
"AttachmentType",
"CalendarContent",
"CalendarDeleteContent",
"CalendarInviteContent",
"CalendarUpdateContent",
"ScheduleItemMetadata",
"ScheduleItemMetadataAttachment",
"ScheduleItemSourceType",
"ScheduleItemStatus",
"parse_calendar_content",
"ScheduleItemCreateRequest",
"ScheduleItemUpdateRequest",
"ScheduleItemResponse",
"ScheduleItemListItem",
"ScheduleItemListRequest",
"ScheduleItemShareRequest",
"ScheduleItemShareResponse",
]
2026-02-28 11:03:29 +08:00
class ScheduleItemCreateRequest(BaseModel):
model_config: ClassVar[ConfigDict] = ConfigDict(extra="forbid")
title: str = Field(min_length=1, max_length=255)
description: str | None = Field(default=None, max_length=2000)
start_at: datetime
end_at: datetime | None = None
timezone: str = Field(..., min_length=1, max_length=50)
2026-02-28 11:03:29 +08:00
metadata: ScheduleItemMetadata | None = None
@field_validator("timezone")
@classmethod
def validate_timezone(cls, value: str) -> str:
try:
ZoneInfo(value)
except ZoneInfoNotFoundError as exc:
raise ValueError("timezone must be a valid IANA timezone") from exc
return value
@field_validator("start_at", "end_at")
@classmethod
def validate_datetime_tzinfo(cls, value: datetime | None) -> datetime | None:
if value is None:
return None
if value.tzinfo is None:
raise ValueError("datetime must include timezone offset")
return value
2026-02-28 11:03:29 +08:00
class ScheduleItemUpdateRequest(BaseModel):
model_config: ClassVar[ConfigDict] = ConfigDict(extra="forbid")
title: str | None = Field(default=None, min_length=1, max_length=255)
description: str | None = Field(default=None, max_length=2000)
start_at: datetime | None = None
end_at: datetime | None = None
timezone: str | None = Field(default=None, max_length=50)
2026-02-28 11:03:29 +08:00
metadata: ScheduleItemMetadata | None = None
status: ScheduleItemStatus | None = None
@field_validator("timezone")
@classmethod
def validate_timezone(cls, value: str | None) -> str | None:
if value is None:
return None
try:
ZoneInfo(value)
except ZoneInfoNotFoundError as exc:
raise ValueError("timezone must be a valid IANA timezone") from exc
return value
@field_validator("start_at", "end_at")
@classmethod
def validate_datetime_tzinfo(cls, value: datetime | None) -> datetime | None:
if value is None:
return None
if value.tzinfo is None:
raise ValueError("datetime must include timezone offset")
return value
2026-02-28 11:03:29 +08:00
class ScheduleItemResponse(BaseModel):
model_config: ClassVar[ConfigDict] = ConfigDict(from_attributes=True)
id: UUID
owner_id: UUID
2026-02-28 11:03:29 +08:00
title: str
description: str | None = None
start_at: datetime
end_at: datetime | None = None
timezone: str
metadata: ScheduleItemMetadata | None = None
status: ScheduleItemStatus
source_type: ScheduleItemSourceType
created_at: datetime
updated_at: datetime
permission: int = 1
is_owner: bool = False
2026-02-28 11:03:29 +08:00
class ScheduleItemListItem(BaseModel):
model_config: ClassVar[ConfigDict] = ConfigDict(from_attributes=True)
id: UUID
title: str
start_at: datetime
end_at: datetime | None = None
timezone: str
status: ScheduleItemStatus
class ScheduleItemListRequest(BaseModel):
start_at: datetime
end_at: datetime
2026-02-28 12:15:59 +08:00
@field_validator("start_at", "end_at")
@classmethod
def validate_datetime_tzinfo(cls, value: datetime) -> datetime:
if value.tzinfo is None:
raise ValueError("datetime must include timezone offset")
return value
2026-02-28 12:15:59 +08:00
_PERMISSION_VIEW = 1
_PERMISSION_INVITE = 2
_PERMISSION_EDIT = 4
2026-02-28 12:15:59 +08:00
class ScheduleItemShareRequest(BaseModel):
model_config: ClassVar[ConfigDict] = ConfigDict(extra="forbid")
phone: str = Field(
...,
pattern=r"^\+861[3-9]\d{9}$",
description="Phone of user to share with",
)
2026-02-28 12:15:59 +08:00
permission_view: bool = Field(True, description="Grant view permission")
permission_edit: bool = Field(False, description="Grant edit permission")
permission_invite: bool = Field(False, description="Grant invite permission")
def _permission_value(self) -> int:
value = 0
if self.permission_view:
value |= _PERMISSION_VIEW
2026-02-28 12:15:59 +08:00
if self.permission_edit:
value |= _PERMISSION_EDIT
2026-02-28 12:15:59 +08:00
if self.permission_invite:
value |= _PERMISSION_INVITE
2026-02-28 12:15:59 +08:00
return value
class ScheduleItemShareResponse(BaseModel):
message: str