- Add `tags: List[str]` field to all 13 entity types (devices, output targets, CSS sources, picture sources, audio sources, value sources, sync clocks, automations, scene presets, capture/audio/PP/pattern templates) - Update all stores, schemas, and route handlers for tag CRUD - Add GET /api/v1/tags endpoint aggregating unique tags across all stores - Create TagInput component with chip display, autocomplete dropdown, keyboard navigation, and API-backed suggestions - Display tag chips on all entity cards (searchable via existing text filter) - Add tag input to all 14 editor modals with dirty check support - Add CSS styles and i18n keys (en/ru/zh) for tag UI - Also includes code review fixes: thread safety, perf, store dedup Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>
60 lines
3.0 KiB
Python
60 lines
3.0 KiB
Python
"""Audio source schemas (CRUD)."""
|
|
|
|
from datetime import datetime
|
|
from typing import List, Literal, Optional
|
|
|
|
from pydantic import BaseModel, Field
|
|
|
|
|
|
class AudioSourceCreate(BaseModel):
|
|
"""Request to create an audio source."""
|
|
|
|
name: str = Field(description="Source name", min_length=1, max_length=100)
|
|
source_type: Literal["multichannel", "mono"] = Field(description="Source type")
|
|
# multichannel fields
|
|
device_index: Optional[int] = Field(None, description="Audio device index (-1 = default)")
|
|
is_loopback: Optional[bool] = Field(None, description="True for system audio (WASAPI loopback)")
|
|
audio_template_id: Optional[str] = Field(None, description="Audio capture template ID")
|
|
# mono fields
|
|
audio_source_id: Optional[str] = Field(None, description="Parent multichannel audio source ID")
|
|
channel: Optional[str] = Field(None, description="Channel: mono|left|right")
|
|
description: Optional[str] = Field(None, description="Optional description", max_length=500)
|
|
tags: List[str] = Field(default_factory=list, description="User-defined tags")
|
|
|
|
|
|
class AudioSourceUpdate(BaseModel):
|
|
"""Request to update an audio source."""
|
|
|
|
name: Optional[str] = Field(None, description="Source name", min_length=1, max_length=100)
|
|
device_index: Optional[int] = Field(None, description="Audio device index (-1 = default)")
|
|
is_loopback: Optional[bool] = Field(None, description="True for system audio (WASAPI loopback)")
|
|
audio_template_id: Optional[str] = Field(None, description="Audio capture template ID")
|
|
audio_source_id: Optional[str] = Field(None, description="Parent multichannel audio source ID")
|
|
channel: Optional[str] = Field(None, description="Channel: mono|left|right")
|
|
description: Optional[str] = Field(None, description="Optional description", max_length=500)
|
|
tags: Optional[List[str]] = None
|
|
|
|
|
|
class AudioSourceResponse(BaseModel):
|
|
"""Audio source response."""
|
|
|
|
id: str = Field(description="Source ID")
|
|
name: str = Field(description="Source name")
|
|
source_type: str = Field(description="Source type: multichannel or mono")
|
|
device_index: Optional[int] = Field(None, description="Audio device index")
|
|
is_loopback: Optional[bool] = Field(None, description="WASAPI loopback mode")
|
|
audio_template_id: Optional[str] = Field(None, description="Audio capture template ID")
|
|
audio_source_id: Optional[str] = Field(None, description="Parent multichannel source ID")
|
|
channel: Optional[str] = Field(None, description="Channel: mono|left|right")
|
|
description: Optional[str] = Field(None, description="Description")
|
|
tags: List[str] = Field(default_factory=list, description="User-defined tags")
|
|
created_at: datetime = Field(description="Creation timestamp")
|
|
updated_at: datetime = Field(description="Last update timestamp")
|
|
|
|
|
|
class AudioSourceListResponse(BaseModel):
|
|
"""List of audio sources."""
|
|
|
|
sources: List[AudioSourceResponse] = Field(description="List of audio sources")
|
|
count: int = Field(description="Number of sources")
|