2026-02-28 11:03:29 +08:00
|
|
|
from __future__ import annotations
|
|
|
|
|
|
2026-03-11 20:51:56 +08:00
|
|
|
import json
|
2026-02-28 11:03:29 +08:00
|
|
|
from datetime import datetime
|
|
|
|
|
from enum import Enum
|
2026-03-11 20:51:56 +08:00
|
|
|
from typing import Literal, ClassVar, Union
|
2026-02-28 11:03:29 +08:00
|
|
|
from uuid import UUID
|
|
|
|
|
|
2026-02-28 12:40:40 +08:00
|
|
|
from pydantic import BaseModel, ConfigDict, EmailStr, Field
|
2026-02-28 11:03:29 +08:00
|
|
|
|
|
|
|
|
|
|
|
|
|
class AttachmentType(str, Enum):
|
|
|
|
|
DOCUMENT = "document"
|
|
|
|
|
REMINDER = "reminder"
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
class ScheduleItemMetadataAttachment(BaseModel):
|
2026-03-11 15:28:29 +08:00
|
|
|
model_config: ClassVar[ConfigDict] = ConfigDict(extra="forbid")
|
|
|
|
|
|
2026-02-28 11:03:29 +08:00
|
|
|
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):
|
2026-03-11 15:28:29 +08:00
|
|
|
model_config: ClassVar[ConfigDict] = ConfigDict(extra="forbid")
|
|
|
|
|
|
|
|
|
|
color: str | None = Field(default=None, pattern=r"^#[0-9A-Fa-f]{6}$")
|
2026-02-28 11:03:29 +08:00
|
|
|
location: str | None = None
|
|
|
|
|
notes: str | None = None
|
2026-02-28 11:29:06 +08:00
|
|
|
attachments: list[ScheduleItemMetadataAttachment] = Field(default_factory=list)
|
2026-03-11 17:16:11 +08:00
|
|
|
reminder_minutes: int | None = Field(default=None, ge=0, le=10080)
|
2026-03-11 15:28:29 +08:00
|
|
|
version: Literal[1] = 1
|
2026-02-28 11:03:29 +08:00
|
|
|
|
|
|
|
|
|
|
|
|
|
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
|
2026-03-11 20:51:56 +08:00
|
|
|
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
|
2026-03-11 20:51:56 +08:00
|
|
|
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
|
|
|
|
|
|
|
|
|
2026-02-28 12:48:01 +08:00
|
|
|
# Permission bit constants (matching PermissionBits in inbox_messages/schemas.py)
|
|
|
|
|
_PERMISSION_VIEW = 1 # 001
|
|
|
|
|
_PERMISSION_INVITE = 2 # 010
|
|
|
|
|
_PERMISSION_EDIT = 4 # 100
|
|
|
|
|
|
|
|
|
|
|
2026-02-28 12:15:59 +08:00
|
|
|
class ScheduleItemShareRequest(BaseModel):
|
2026-02-28 12:40:40 +08:00
|
|
|
model_config: ClassVar[ConfigDict] = ConfigDict(extra="forbid")
|
|
|
|
|
|
|
|
|
|
email: EmailStr = Field(..., description="Email 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:
|
2026-02-28 12:48:01 +08:00
|
|
|
value |= _PERMISSION_VIEW
|
2026-02-28 12:15:59 +08:00
|
|
|
if self.permission_edit:
|
2026-02-28 12:48:01 +08:00
|
|
|
value |= _PERMISSION_EDIT
|
2026-02-28 12:15:59 +08:00
|
|
|
if self.permission_invite:
|
2026-02-28 12:48:01 +08:00
|
|
|
value |= _PERMISSION_INVITE
|
2026-02-28 12:15:59 +08:00
|
|
|
return value
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
class ScheduleItemShareResponse(BaseModel):
|
|
|
|
|
message: str
|
2026-03-11 20:51:56 +08:00
|
|
|
|
|
|
|
|
|
|
|
|
|
class CalendarInviteContent(BaseModel):
|
|
|
|
|
model_config: ClassVar[ConfigDict] = ConfigDict(extra="forbid")
|
|
|
|
|
|
|
|
|
|
type: Literal["invite"]
|
|
|
|
|
permission: int = Field(..., description="权限: 1=view, 4=edit, 8=invite")
|
|
|
|
|
action: Literal["pending"] = "pending"
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
class CalendarUpdateContent(BaseModel):
|
|
|
|
|
model_config: ClassVar[ConfigDict] = ConfigDict(extra="forbid")
|
|
|
|
|
|
|
|
|
|
type: Literal["update"]
|
|
|
|
|
title: str = Field(..., description="事件标题")
|
|
|
|
|
action: Literal["updated"] = "updated"
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
class CalendarDeleteContent(BaseModel):
|
|
|
|
|
model_config: ClassVar[ConfigDict] = ConfigDict(extra="forbid")
|
|
|
|
|
|
|
|
|
|
type: Literal["delete"]
|
|
|
|
|
title: str = Field(..., description="事件标题")
|
|
|
|
|
action: Literal["deleted"] = "deleted"
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
CalendarContent = Union[
|
|
|
|
|
CalendarInviteContent, CalendarUpdateContent, CalendarDeleteContent
|
|
|
|
|
]
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
def parse_calendar_content(content: str | None) -> CalendarContent | None:
|
|
|
|
|
if not content:
|
|
|
|
|
return None
|
|
|
|
|
try:
|
|
|
|
|
data = json.loads(content)
|
|
|
|
|
content_type = data.get("type")
|
|
|
|
|
if content_type == "invite":
|
|
|
|
|
return CalendarInviteContent(**data)
|
|
|
|
|
elif content_type == "update":
|
|
|
|
|
return CalendarUpdateContent(**data)
|
|
|
|
|
elif content_type == "delete":
|
|
|
|
|
return CalendarDeleteContent(**data)
|
|
|
|
|
else:
|
|
|
|
|
raise ValueError(f"Unknown calendar content type: {content_type}")
|
|
|
|
|
except Exception:
|
|
|
|
|
return None
|