kiwi/app/models/schemas/inventory.py
pyr0ball 890216a1f0
Some checks are pending
CI / Backend (Python) (push) Waiting to run
CI / Frontend (Vue) (push) Waiting to run
Mirror / mirror (push) Waiting to run
fix: wire recipe corpus to cloud per-user DBs via SQLite ATTACH (#102)
Cloud mode: attach shared read-only corpus DB (RECIPE_DB_PATH env var)
as "corpus" schema so per-user SQLite DBs can access 3.19M recipes.
All corpus table references now use self._cp prefix ("corpus." in cloud,
"" in local). FTS5 pseudo-column kept unqualified per SQLite spec.
compose.cloud.yml: bind-mount /Library/Assets/kiwi/kiwi.db read-only.

Also fix batch of audit issues:
- #101: OCR approval used source="receipt_ocr" for inventory_items — use "receipt"
- #89/#100: Shopping confirm-purchase used source="shopping_list" — use "manual"
- #103: Frontend inventory filter sent ?status= but API expects ?item_status=
- #104: InventoryItemUpdate schema missing purchase_date field; store.py allowed set also missing it
- #105: Guest cookie Secure flag tied to CLOUD_MODE instead of X-Forwarded-Proto; broke HTTP direct-port access
2026-04-18 14:21:56 -07:00

187 lines
5.7 KiB
Python

"""Pydantic schemas for inventory management (integer IDs, SQLite-compatible)."""
from __future__ import annotations
from datetime import date, datetime
from typing import Any, Dict, List, Optional
from pydantic import BaseModel, Field
# ── Tags ──────────────────────────────────────────────────────────────────────
class TagCreate(BaseModel):
name: str = Field(..., max_length=100)
slug: str = Field(..., max_length=100)
description: Optional[str] = None
color: Optional[str] = Field(None, max_length=7)
category: Optional[str] = None
class TagResponse(BaseModel):
id: int
name: str
slug: str
description: Optional[str]
color: Optional[str]
category: Optional[str]
created_at: str
updated_at: str
model_config = {"from_attributes": True}
# ── Products ──────────────────────────────────────────────────────────────────
class ProductCreate(BaseModel):
name: str = Field(..., max_length=500)
barcode: Optional[str] = Field(None, max_length=50)
brand: Optional[str] = None
category: Optional[str] = None
description: Optional[str] = None
image_url: Optional[str] = None
nutrition_data: Dict[str, Any] = Field(default_factory=dict)
source: str = "manual"
source_data: Dict[str, Any] = Field(default_factory=dict)
class ProductUpdate(BaseModel):
name: Optional[str] = None
brand: Optional[str] = None
category: Optional[str] = None
description: Optional[str] = None
image_url: Optional[str] = None
nutrition_data: Optional[Dict[str, Any]] = None
class ProductResponse(BaseModel):
id: int
barcode: Optional[str]
name: str
brand: Optional[str]
category: Optional[str]
description: Optional[str]
image_url: Optional[str]
nutrition_data: Dict[str, Any]
source: str
created_at: str
updated_at: str
model_config = {"from_attributes": True}
# ── Inventory Items ───────────────────────────────────────────────────────────
class InventoryItemCreate(BaseModel):
product_id: int
quantity: float = Field(default=1.0, gt=0)
unit: str = "count"
location: str
sublocation: Optional[str] = None
purchase_date: Optional[date] = None
expiration_date: Optional[date] = None
notes: Optional[str] = None
source: str = "manual"
class InventoryItemUpdate(BaseModel):
quantity: Optional[float] = Field(None, gt=0)
unit: Optional[str] = None
location: Optional[str] = None
sublocation: Optional[str] = None
purchase_date: Optional[date] = None
expiration_date: Optional[date] = None
opened_date: Optional[date] = None
status: Optional[str] = None
notes: Optional[str] = None
disposal_reason: Optional[str] = None
class PartialConsumeRequest(BaseModel):
quantity: float = Field(..., gt=0, description="Amount to consume from this item")
class DiscardRequest(BaseModel):
reason: Optional[str] = Field(None, max_length=200)
class InventoryItemResponse(BaseModel):
id: int
product_id: int
product_name: Optional[str] = None
barcode: Optional[str] = None
category: Optional[str] = None
quantity: float
unit: str
location: str
sublocation: Optional[str]
purchase_date: Optional[str]
expiration_date: Optional[str]
opened_date: Optional[str] = None
opened_expiry_date: Optional[str] = None
secondary_state: Optional[str] = None
secondary_uses: Optional[List[str]] = None
secondary_warning: Optional[str] = None
status: str
notes: Optional[str]
disposal_reason: Optional[str] = None
source: str
created_at: str
updated_at: str
model_config = {"from_attributes": True}
# ── Barcode scan ──────────────────────────────────────────────────────────────
class BarcodeScanResult(BaseModel):
barcode: str
barcode_type: str
product: Optional[ProductResponse]
inventory_item: Optional[InventoryItemResponse]
added_to_inventory: bool
needs_manual_entry: bool = False
message: str
class BarcodeScanResponse(BaseModel):
success: bool
barcodes_found: int
results: List[BarcodeScanResult]
message: str
# ── Bulk add by name ─────────────────────────────────────────────────────────
class BulkAddItem(BaseModel):
name: str = Field(..., min_length=1, max_length=200)
quantity: float = Field(default=1.0, gt=0)
unit: str = "count"
location: str = "pantry"
class BulkAddByNameRequest(BaseModel):
items: List[BulkAddItem] = Field(..., min_length=1)
class BulkAddItemResult(BaseModel):
name: str
ok: bool
item_id: Optional[int] = None
error: Optional[str] = None
class BulkAddByNameResponse(BaseModel):
added: int
failed: int
results: List[BulkAddItemResult]
# ── Stats ─────────────────────────────────────────────────────────────────────
class InventoryStats(BaseModel):
total_items: int
available_items: int
expiring_soon: int
expired_items: int
locations: Dict[str, int]