Implements the full User Connections & Notification Centre feature: Phase 1 - Database: migrations 039-043 adding umbral_name to users, profile/social fields to settings, notifications table, connection request/user_connection tables, and linked_user_id to people. Phase 2 - Notifications: backend CRUD router + service + 90-day purge, frontend NotificationsPage with All/Unread filter, bell icon in sidebar with unread badge polling every 60s. Phase 3 - Settings: profile fields (phone, mobile, address, company, job_title), social card with accept_connections toggle and per-field sharing defaults, umbral name display with CopyableField. Phase 4 - Connections: timing-safe user search, send/accept/reject flow with atomic status updates, bidirectional UserConnection + Person records, in-app + ntfy notifications, per-receiver pending cap, nginx rate limiting. Phase 5 - People integration: batch-loaded shared profiles (N+1 prevention), Ghost icon for umbral contacts, Umbral filter pill, split Add Person button, shared field indicators (synced labels + Lock icons), disabled form inputs for synced fields on umbral contacts. Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>
31 lines
746 B
Python
31 lines
746 B
Python
from pydantic import BaseModel, ConfigDict, Field
|
|
from datetime import datetime
|
|
from typing import Optional
|
|
|
|
|
|
class NotificationResponse(BaseModel):
|
|
id: int
|
|
user_id: int
|
|
type: str
|
|
title: Optional[str] = None
|
|
message: Optional[str] = None
|
|
data: Optional[dict] = None
|
|
source_type: Optional[str] = None
|
|
source_id: Optional[int] = None
|
|
is_read: bool
|
|
created_at: datetime
|
|
|
|
model_config = ConfigDict(from_attributes=True)
|
|
|
|
|
|
class NotificationListResponse(BaseModel):
|
|
notifications: list[NotificationResponse]
|
|
unread_count: int
|
|
total: int
|
|
|
|
|
|
class MarkReadRequest(BaseModel):
|
|
model_config = ConfigDict(extra="forbid")
|
|
|
|
notification_ids: list[int] = Field(..., min_length=1, max_length=100)
|