Files
platform/services/brain/app/models/schema.py
Yusuf Suleman 4592e35732
All checks were successful
Security Checks / dependency-audit (push) Successful in 1m13s
Security Checks / secret-scanning (push) Successful in 3s
Security Checks / dockerfile-lint (push) Successful in 3s
feat: major platform expansion — Brain service, RSS reader, iOS app, AI assistants, Firefox extension
Brain Service:
- Playwright stealth crawler replacing browserless (og:image, Readability, Reddit JSON API)
- AI classification with tag definitions and folder assignment
- YouTube video download via yt-dlp
- Karakeep migration complete (96 items)
- Taxonomy management (folders with icons/colors, tags)
- Discovery shuffle, sort options, search (Meilisearch + pgvector)
- Item tag/folder editing, card color accents

RSS Reader Service:
- Custom FastAPI reader replacing Miniflux
- Feed management (add/delete/refresh), category support
- Full article extraction via Readability
- Background content fetching for new entries
- Mark all read with confirmation
- Infinite scroll, retention cleanup (30/60 day)
- 17 feeds migrated from Miniflux

iOS App (SwiftUI):
- Native iOS 17+ app with @Observable architecture
- Cookie-based auth, configurable gateway URL
- Dashboard with custom background photo + frosted glass widgets
- Full fitness module (today/templates/goals/food library)
- AI assistant chat (fitness + brain, raw JSON state management)
- 120fps ProMotion support

AI Assistants (Gateway):
- Unified dispatcher with fitness/brain domain detection
- Fitness: natural language food logging, photo analysis, multi-item splitting
- Brain: save/append/update/delete notes, search & answer, undo support
- Madiha user gets fitness-only (brain disabled)

Firefox Extension:
- One-click save to Brain from any page
- Login with platform credentials
- Right-click context menu (save page/link/image)
- Notes field for URL saves
- Signed and published on AMO

Other:
- Reader bookmark button routes to Brain (was Karakeep)
- Fitness food library with "Add" button + add-to-meal popup
- Kindle send file size check (25MB SMTP2GO limit)
- Atelier UI as default (useAtelierShell=true)
- Mobile upload box in nav drawer

Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
2026-04-03 00:56:29 -05:00

132 lines
2.9 KiB
Python

"""Pydantic schemas for API request/response."""
from __future__ import annotations
from datetime import datetime
from typing import Optional
from pydantic import BaseModel, Field
# ── Request schemas ──
class ItemCreate(BaseModel):
type: str = "link"
url: Optional[str] = None
raw_content: Optional[str] = None
title: Optional[str] = None
folder: Optional[str] = None
tags: Optional[list[str]] = None
class ItemUpdate(BaseModel):
title: Optional[str] = None
folder: Optional[str] = None
tags: Optional[list[str]] = None
raw_content: Optional[str] = None
class ItemAdditionCreate(BaseModel):
content: str
source: Optional[str] = "assistant"
kind: Optional[str] = "append"
metadata_json: Optional[dict] = None
class SearchQuery(BaseModel):
q: str
folder: Optional[str] = None
tags: Optional[list[str]] = None
type: Optional[str] = None
limit: int = Field(default=20, le=100)
offset: int = 0
class SemanticSearchQuery(BaseModel):
q: str
folder: Optional[str] = None
type: Optional[str] = None
limit: int = Field(default=20, le=100)
class HybridSearchQuery(BaseModel):
q: str
folder: Optional[str] = None
tags: Optional[list[str]] = None
type: Optional[str] = None
limit: int = Field(default=20, le=100)
# ── Response schemas ──
class AssetOut(BaseModel):
id: str
asset_type: str
filename: str
content_type: Optional[str] = None
size_bytes: Optional[int] = None
created_at: datetime
model_config = {"from_attributes": True}
class ItemAdditionOut(BaseModel):
id: str
item_id: str
source: str
kind: str
content: str
metadata_json: Optional[dict] = None
created_at: datetime
updated_at: datetime
model_config = {"from_attributes": True}
class ItemOut(BaseModel):
id: str
type: str
title: Optional[str] = None
url: Optional[str] = None
raw_content: Optional[str] = None
extracted_text: Optional[str] = None
folder: Optional[str] = None
tags: Optional[list[str]] = None
summary: Optional[str] = None
confidence: Optional[float] = None
processing_status: str
processing_error: Optional[str] = None
metadata_json: Optional[dict] = None
created_at: datetime
updated_at: datetime
assets: list[AssetOut] = []
model_config = {"from_attributes": True}
class ItemList(BaseModel):
items: list[ItemOut]
total: int
class SearchResult(BaseModel):
items: list[ItemOut]
total: int
query: str
class ConfigOut(BaseModel):
folders: list[str]
tags: list[str]
# ── OpenAI classification schema ──
class ClassificationResult(BaseModel):
"""What the AI returns for each item."""
folder: str
tags: list[str] = Field(min_length=2, max_length=3)
title: str
summary: str
confidence: float = Field(ge=0.0, le=1.0)