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>
132 lines
2.9 KiB
Python
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)
|