refactor: Jitsi/Matrix/Voice von Core übernommen, Camunda/BPMN gelöscht, Kommunikation-Nav
Some checks failed
CI / go-lint (push) Has been skipped
CI / python-lint (push) Has been skipped
CI / nodejs-lint (push) Has been skipped
CI / test-go-school (push) Successful in 25s
CI / test-go-edu-search (push) Successful in 26s
CI / test-python-klausur (push) Failing after 1m55s
CI / test-python-agent-core (push) Successful in 16s
CI / test-nodejs-website (push) Successful in 18s

- Voice-Service von Core nach Lehrer verschoben (bp-lehrer-voice-service)
- 4 Jitsi-Services + 2 Synapse-Services in docker-compose.yml aufgenommen
- Camunda komplett gelöscht: workflow pages, workflow-config.ts, bpmn-js deps
- CAMUNDA_URL aus backend-lehrer environment entfernt
- Sidebar: Kategorie "Compliance SDK" + "Katalogverwaltung" entfernt
- Sidebar: Neue Kategorie "Kommunikation" mit Video & Chat, Voice Service, Alerts

Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>
This commit is contained in:
Benjamin Admin
2026-03-05 17:01:47 +01:00
parent 2ec4d8aabd
commit 9912997187
68 changed files with 12992 additions and 1432 deletions

117
voice-service/config.py Normal file
View File

@@ -0,0 +1,117 @@
"""
Voice Service Configuration
Environment-based configuration with Pydantic Settings
DSGVO-konform: Keine Audio-Persistenz, nur transiente Verarbeitung
"""
from functools import lru_cache
from typing import Optional, List
from pydantic_settings import BaseSettings, SettingsConfigDict
class Settings(BaseSettings):
"""Application settings loaded from environment variables."""
model_config = SettingsConfigDict(
env_file=".env",
env_file_encoding="utf-8",
case_sensitive=False,
extra="ignore", # Ignore unknown environment variables from docker-compose
)
# Service Config
port: int = 8091
environment: str = "development"
debug: bool = False
# JWT Authentication (load from Vault or environment, test default for CI)
jwt_secret: str = "test-secret-for-ci-only-do-not-use-in-production"
jwt_algorithm: str = "HS256"
jwt_expiration_hours: int = 24
# PostgreSQL (load from Vault or environment, test default for CI)
database_url: str = "postgresql://test:test@localhost:5432/test"
# Valkey (Redis-fork) Session Cache
valkey_url: str = "redis://valkey:6379/2"
session_ttl_hours: int = 24
task_ttl_hours: int = 168 # 7 days for pending tasks
# PersonaPlex Configuration (Production GPU)
personaplex_enabled: bool = False
personaplex_ws_url: str = "ws://host.docker.internal:8998"
personaplex_model: str = "personaplex-7b"
personaplex_timeout: int = 30
# Task Orchestrator
orchestrator_enabled: bool = True
orchestrator_max_concurrent_tasks: int = 10
# Fallback LLM (Ollama for Development)
fallback_llm_provider: str = "ollama" # "ollama" or "none"
ollama_base_url: str = "http://host.docker.internal:11434"
ollama_voice_model: str = "qwen2.5:32b"
ollama_timeout: int = 120
# Klausur Service Integration
klausur_service_url: str = "http://klausur-service:8086"
# Audio Configuration
audio_sample_rate: int = 24000 # 24kHz for Mimi codec
audio_frame_size_ms: int = 80 # 80ms frames
audio_persistence: bool = False # NEVER persist audio
# Encryption Configuration
encryption_enabled: bool = True
namespace_key_algorithm: str = "AES-256-GCM"
# TTL Configuration (DSGVO Data Minimization)
transcript_ttl_days: int = 7
task_state_ttl_days: int = 30
audit_log_ttl_days: int = 90
# Rate Limiting
max_sessions_per_user: int = 5
max_requests_per_minute: int = 60
# CORS (for frontend access)
cors_origins: List[str] = [
"http://localhost:3000",
"http://localhost:3001",
"http://localhost:8091",
"http://macmini:3000",
"http://macmini:3001",
"https://localhost",
"https://localhost:3000",
"https://localhost:3001",
"https://localhost:8091",
"https://macmini",
"https://macmini:3000",
"https://macmini:3001",
"https://macmini:8091",
]
@property
def is_development(self) -> bool:
"""Check if running in development mode."""
return self.environment == "development"
@property
def audio_frame_samples(self) -> int:
"""Calculate samples per frame."""
return int(self.audio_sample_rate * self.audio_frame_size_ms / 1000)
@property
def use_personaplex(self) -> bool:
"""Check if PersonaPlex should be used (production only)."""
return self.personaplex_enabled and not self.is_development
@lru_cache
def get_settings() -> Settings:
"""Get cached settings instance."""
return Settings()
# Export settings instance for convenience
settings = get_settings()