Files
Timmy-time-dashboard/docker-compose.yml
Alexander Whitestone 317669a21e
Some checks failed
Tests / lint (pull_request) Failing after 32s
Tests / test (pull_request) Has been skipped
feat: add Mumble voice bridge for Alexander–Timmy co-play audio (#858)
Implements a bidirectional Mumble voice bridge so Alexander and Timmy can
have voice conversations during co-play sessions, with audio piped to the
stream.

Changes:
- src/integrations/mumble/bridge.py — MumbleBridge class with:
  - Connect/disconnect lifecycle (graceful degradation when pymumble-py3 absent)
  - speak(text) — converts Timmy's TTS output to PCM and sends to Mumble
  - send_audio(pcm) — raw 16-bit 48 kHz mono PCM transmit
  - push_to_talk() context manager for PTT mode
  - VAD mode (voice activity detection with configurable RMS threshold)
  - on_audio callbacks for audio received from Alexander's mic
  - Pure-Python resampling / stereo-to-mono (no audioop, Python 3.13+ compat)
  - Piper TTS → pyttsx3 TTS fallback chain
- src/config.py — 9 new Mumble settings (host, port, user, password,
  channel, audio_mode, vad_threshold, silence_ms, enabled)
- pyproject.toml — pymumble-py3 optional dep + mumble extras group
- docker-compose.yml — mumblevoip/mumble-server under mumble profile
  (ports 64738 TCP+UDP; start with: docker compose --profile mumble up)
- tests/integrations/test_mumble_bridge.py — 39 unit tests
- src/integrations/CLAUDE.md — document mumble module

Fixes #858

Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>
2026-03-23 22:18:01 -04:00

195 lines
6.9 KiB
YAML

# ── Development Compose ───────────────────────────────────────────────────────
#
# Services
# dashboard FastAPI app (always on)
# celery-worker (behind 'celery' profile)
# openfang (behind 'openfang' profile)
#
# Usage
# make docker-build build the image
# make docker-up start dashboard
# make docker-down stop everything
# make docker-logs tail logs
#
# ── Security note ─────────────────────────────────────────────────────────
# Override user per-environment — see docker-compose.dev.yml / docker-compose.prod.yml
#
# ── Ollama host access ──────────────────────────────────────────────────────
# By default OLLAMA_URL points to http://host.docker.internal:11434 which
# reaches Ollama running on the Docker host (macOS/Windows native).
#
# Linux: The extra_hosts entry maps host.docker.internal → host-gateway,
# which resolves to the host IP on Docker 20.10+.
services:
# ── Dashboard (FastAPI) ──────────────────────────────────────────────────
dashboard:
build: .
image: timmy-time:latest
container_name: timmy-dashboard
user: "" # see security note above
ports:
- "8000:8000"
volumes:
- timmy-data:/app/data
- ./src:/app/src # live-reload: source changes reflect immediately
- ./static:/app/static # live-reload: CSS/asset changes reflect immediately
environment:
DEBUG: "true"
OLLAMA_URL: "${OLLAMA_URL:-http://host.docker.internal:11434}"
# Grok (xAI) — opt-in premium cloud backend
GROK_ENABLED: "${GROK_ENABLED:-false}"
XAI_API_KEY: "${XAI_API_KEY:-}"
GROK_DEFAULT_MODEL: "${GROK_DEFAULT_MODEL:-grok-3-fast}"
# Search backend (SearXNG + Crawl4AI) — set TIMMY_SEARCH_BACKEND=none to disable
TIMMY_SEARCH_BACKEND: "${TIMMY_SEARCH_BACKEND:-searxng}"
TIMMY_SEARCH_URL: "${TIMMY_SEARCH_URL:-http://searxng:8080}"
TIMMY_CRAWL_URL: "${TIMMY_CRAWL_URL:-http://crawl4ai:11235}"
extra_hosts:
- "host.docker.internal:host-gateway" # Linux: maps to host IP
networks:
- timmy-net
restart: unless-stopped
healthcheck:
test: ["CMD", "curl", "-f", "http://localhost:8000/health"]
interval: 30s
timeout: 5s
retries: 3
start_period: 30s
# ── Celery Worker — background task processing ──────────────────────────
celery-worker:
build: .
image: timmy-time:latest
container_name: timmy-celery-worker
user: ""
command: ["celery", "-A", "infrastructure.celery.app", "worker", "--loglevel=info", "--concurrency=2"]
volumes:
- timmy-data:/app/data
- ./src:/app/src
environment:
OLLAMA_URL: "${OLLAMA_URL:-http://host.docker.internal:11434}"
extra_hosts:
- "host.docker.internal:host-gateway"
networks:
- timmy-net
restart: unless-stopped
profiles:
- celery
# ── SearXNG — self-hosted meta-search engine ─────────────────────────
searxng:
image: searxng/searxng:latest
container_name: timmy-searxng
profiles:
- search
ports:
- "${SEARXNG_PORT:-8888}:8080"
environment:
SEARXNG_BASE_URL: "${SEARXNG_BASE_URL:-http://localhost:8888}"
volumes:
- ./docker/searxng:/etc/searxng:rw
networks:
- timmy-net
restart: unless-stopped
healthcheck:
test: ["CMD", "wget", "-qO-", "http://localhost:8080/healthz"]
interval: 30s
timeout: 5s
retries: 3
start_period: 20s
# ── Crawl4AI — self-hosted web scraper ────────────────────────────────
crawl4ai:
image: unclecode/crawl4ai:latest
container_name: timmy-crawl4ai
profiles:
- search
ports:
- "${CRAWL4AI_PORT:-11235}:11235"
environment:
CRAWL4AI_API_TOKEN: "${CRAWL4AI_API_TOKEN:-}"
volumes:
- timmy-data:/app/data
networks:
- timmy-net
restart: unless-stopped
healthcheck:
test: ["CMD", "curl", "-f", "http://localhost:11235/health"]
interval: 30s
timeout: 10s
retries: 3
start_period: 30s
# ── Mumble — voice chat server for Alexander + Timmy ─────────────────────
mumble:
image: mumblevoip/mumble-server:latest
container_name: timmy-mumble
profiles:
- mumble
ports:
- "${MUMBLE_PORT:-64738}:64738" # TCP + UDP: Mumble protocol
- "${MUMBLE_PORT:-64738}:64738/udp"
environment:
MUMBLE_CONFIG_WELCOMETEXT: "Timmy Time voice channel — co-play audio bridge"
MUMBLE_CONFIG_USERS: "10"
MUMBLE_CONFIG_BANDWIDTH: "72000"
# Set MUMBLE_SUPERUSER_PASSWORD in .env to secure the server
MUMBLE_SUPERUSER_PASSWORD: "${MUMBLE_SUPERUSER_PASSWORD:-changeme}"
volumes:
- mumble-data:/data
networks:
- timmy-net
restart: unless-stopped
healthcheck:
test: ["CMD", "sh", "-c", "nc -z localhost 64738 || exit 1"]
interval: 30s
timeout: 5s
retries: 3
start_period: 10s
# ── OpenFang — vendored agent runtime sidecar ────────────────────────────
openfang:
build:
context: .
dockerfile: docker/Dockerfile.openfang
image: timmy-openfang:latest
container_name: timmy-openfang
profiles:
- openfang
environment:
OLLAMA_URL: "${OLLAMA_URL:-http://host.docker.internal:11434}"
OPENFANG_DATA_DIR: "/app/data"
extra_hosts:
- "host.docker.internal:host-gateway"
volumes:
- openfang-data:/app/data
networks:
- timmy-net
restart: unless-stopped
healthcheck:
test: ["CMD", "curl", "-f", "http://localhost:8080/health"]
interval: 30s
timeout: 5s
retries: 3
start_period: 15s
# ── Volumes ──────────────────────────────────────────────────────────────────
volumes:
timmy-data:
driver: local
driver_opts:
type: none
o: bind
device: "${PWD}/data"
openfang-data:
driver: local
mumble-data:
driver: local
# ── Internal network ────────────────────────────────────────────────────────
networks:
timmy-net:
driver: bridge