c69c2ad532
- Remove caller names from Whisper hint (was corrupting transcriptions) - Background gen switched to Claude Sonnet 4.6 (cheap models = thin backgrounds) - "WHAT MAKES A GOOD CALLER" rewritten with concrete examples - Grok guardrails loosened (were cutting too much edge) - Response length guidance added to caller prompt - Retry under-20-word responses once for more detail - Devon monitor softened from "default silence" to balanced - Ban stalling phrases: "where was I", "as I was saying", etc. Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
68 lines
2.7 KiB
Python
68 lines
2.7 KiB
Python
"""Configuration settings for the AI Radio Show backend"""
|
|
|
|
import os
|
|
from pathlib import Path
|
|
from pydantic_settings import BaseSettings
|
|
from dotenv import load_dotenv
|
|
|
|
# Load .env from parent directory
|
|
load_dotenv(Path(__file__).parent.parent / ".env")
|
|
|
|
|
|
class Settings(BaseSettings):
|
|
# API Keys
|
|
elevenlabs_api_key: str = os.getenv("ELEVENLABS_API_KEY", "")
|
|
openrouter_api_key: str = os.getenv("OPENROUTER_API_KEY", "")
|
|
inworld_api_key: str = os.getenv("INWORLD_API_KEY", "")
|
|
|
|
# SignalWire
|
|
signalwire_project_id: str = os.getenv("SIGNALWIRE_PROJECT_ID", "")
|
|
signalwire_space: str = os.getenv("SIGNALWIRE_SPACE", "")
|
|
signalwire_token: str = os.getenv("SIGNALWIRE_TOKEN", "")
|
|
signalwire_phone: str = os.getenv("SIGNALWIRE_PHONE", "")
|
|
signalwire_stream_url: str = os.getenv("SIGNALWIRE_STREAM_URL", "")
|
|
|
|
# Email (IMAP)
|
|
submissions_imap_host: str = os.getenv("SUBMISSIONS_IMAP_HOST", "")
|
|
submissions_imap_user: str = os.getenv("SUBMISSIONS_IMAP_USER", "")
|
|
submissions_imap_pass: str = os.getenv("SUBMISSIONS_IMAP_PASS", "")
|
|
|
|
# LLM Settings
|
|
llm_provider: str = "openrouter" # "openrouter" or "ollama"
|
|
openrouter_model: str = "anthropic/claude-sonnet-4.6" # primary/default model
|
|
ollama_model: str = "llama3.2"
|
|
ollama_host: str = "http://localhost:11434"
|
|
|
|
# Per-category model routing
|
|
# caller_dialog is overridden by style_matched routing (see Session.caller_model_map)
|
|
category_models: dict = {
|
|
"caller_dialog": "x-ai/grok-4.1-fast", # fallback if style_matched disabled ($0.20/$0.50)
|
|
"devon_ask": "x-ai/grok-4.1-fast", # Devon matches show energy, cheap ($0.20/$0.50)
|
|
"devon_monitor": "google/gemini-2.5-flash", # just yes/no decisions, keep cheap ($0.15/$0.60)
|
|
"background_gen": "anthropic/claude-sonnet-4.6", # backgrounds drive the whole call — worth the quality ($3/$15, ~$0.30/show)
|
|
"call_summary": "google/gemini-2.5-flash", # post-call, no personality needed ($0.15/$0.60)
|
|
"news_summary": "google/gemini-2.5-flash", # just digesting headlines ($0.15/$0.60)
|
|
"topic_gen": "google/gemini-2.5-flash", # structured output ($0.15/$0.60)
|
|
}
|
|
|
|
# TTS Settings
|
|
tts_provider: str = "inworld" # "kokoro", "elevenlabs", "inworld", "vits", or "bark"
|
|
|
|
# Audio Settings
|
|
sample_rate: int = 24000
|
|
|
|
# Paths
|
|
base_dir: Path = Path(__file__).parent.parent
|
|
sounds_dir: Path = base_dir / "sounds"
|
|
music_dir: Path = base_dir / "music"
|
|
ads_dir: Path = base_dir / "ads"
|
|
idents_dir: Path = base_dir / "idents"
|
|
sessions_dir: Path = base_dir / "sessions"
|
|
|
|
class Config:
|
|
env_file = ".env"
|
|
extra = "ignore"
|
|
|
|
|
|
settings = Settings()
|