2026-02-28 11:03:29 +08:00
|
|
|
from __future__ import annotations
|
|
|
|
|
|
|
|
|
|
from datetime import datetime
|
|
|
|
|
from enum import Enum
|
|
|
|
|
from typing import ClassVar
|
|
|
|
|
from uuid import UUID
|
|
|
|
|
|
2026-02-28 11:29:06 +08:00
|
|
|
from pydantic import BaseModel, ConfigDict, Field
|
2026-02-28 11:03:29 +08:00
|
|
|
|
|
|
|
|
|
|
|
|
|
class AttachmentType(str, Enum):
|
|
|
|
|
DOCUMENT = "document"
|
|
|
|
|
REMINDER = "reminder"
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
class ScheduleItemMetadataAttachment(BaseModel):
|
|
|
|
|
name: str
|
|
|
|
|
type: AttachmentType
|
2026-02-28 11:29:06 +08:00
|
|
|
visible_to: list[UUID] = Field(default_factory=list)
|
2026-02-28 11:03:29 +08:00
|
|
|
url: str | None = None
|
|
|
|
|
note: str | None = None
|
|
|
|
|
content: str | None = None
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
class ScheduleItemMetadata(BaseModel):
|
|
|
|
|
color: str | None = None
|
|
|
|
|
location: str | None = None
|
|
|
|
|
notes: str | None = None
|
2026-02-28 11:29:06 +08:00
|
|
|
attachments: list[ScheduleItemMetadataAttachment] = Field(default_factory=list)
|
2026-02-28 11:03:29 +08:00
|
|
|
version: int = 1
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
class ScheduleItemStatus(str, Enum):
|
|
|
|
|
ACTIVE = "active"
|
|
|
|
|
COMPLETED = "completed"
|
|
|
|
|
CANCELED = "canceled"
|
|
|
|
|
ARCHIVED = "archived"
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
class ScheduleItemSourceType(str, Enum):
|
|
|
|
|
MANUAL = "manual"
|
|
|
|
|
IMPORTED = "imported"
|
|
|
|
|
AGENT_GENERATED = "agent_generated"
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
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
|
2026-02-28 11:29:06 +08:00
|
|
|
timezone: str = Field(default="UTC", max_length=50)
|
2026-02-28 11:03:29 +08:00
|
|
|
metadata: ScheduleItemMetadata | None = None
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
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
|
2026-02-28 11:29:06 +08:00
|
|
|
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
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
class ScheduleItemResponse(BaseModel):
|
|
|
|
|
model_config: ClassVar[ConfigDict] = ConfigDict(from_attributes=True)
|
|
|
|
|
|
|
|
|
|
id: UUID
|
|
|
|
|
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
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
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
|
|
|
|
|
|
|
|
|
|
|
|
|
class ScheduleItemShareRequest(BaseModel):
|
|
|
|
|
email: str = Field(..., description="Email of user to share with")
|
|
|
|
|
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 |= 1
|
|
|
|
|
if self.permission_edit:
|
|
|
|
|
value |= 4
|
|
|
|
|
if self.permission_invite:
|
|
|
|
|
value |= 2
|
|
|
|
|
return value
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
class ScheduleItemShareResponse(BaseModel):
|
|
|
|
|
message: str
|