feat(microphone): PollMonitor mit Geräteerkennung (TDD)

This commit is contained in:
2026-05-14 17:37:24 +02:00
parent 02496fb708
commit de6c61aeb3
2 changed files with 123 additions and 0 deletions
+67
View File
@@ -0,0 +1,67 @@
# tests/test_microphone_monitor.py
import asyncio
from unittest.mock import AsyncMock, patch
import pytest
from whisper_local.microphone._poll import PollMonitor
def _fake_devices(names: list[str]) -> list[dict]:
return [{"name": n, "max_input_channels": 1} for n in names]
@pytest.mark.asyncio
async def test_on_device_added_fires_when_device_appears():
monitor = PollMonitor(configured_device=None, interval=0.05)
event = asyncio.Event()
added: list[str] = []
async def on_added(name: str) -> None:
added.append(name)
event.set()
monitor.on_device_added = on_added
call_count = 0
def fake_query():
nonlocal call_count
call_count += 1
if call_count == 1:
return _fake_devices(["Mic A"])
return _fake_devices(["Mic A", "Mic B"])
with patch("sounddevice.query_devices", side_effect=fake_query):
await monitor.start()
await asyncio.wait_for(event.wait(), timeout=1.0)
monitor.stop()
assert added == ["Mic B"]
@pytest.mark.asyncio
async def test_on_device_removed_fires_when_device_disappears():
monitor = PollMonitor(configured_device=None, interval=0.05)
event = asyncio.Event()
removed: list[str] = []
async def on_removed(name: str) -> None:
removed.append(name)
event.set()
monitor.on_device_removed = on_removed
call_count = 0
def fake_query():
nonlocal call_count
call_count += 1
if call_count == 1:
return _fake_devices(["Mic A", "Mic B"])
return _fake_devices(["Mic A"])
with patch("sounddevice.query_devices", side_effect=fake_query):
await monitor.start()
await asyncio.wait_for(event.wait(), timeout=1.0)
monitor.stop()
assert removed == ["Mic B"]
+56
View File
@@ -0,0 +1,56 @@
# whisper_local/microphone/_poll.py
"""Polling-basierter Mikrofon-Monitor (cross-platform)."""
import asyncio
import logging
from collections.abc import Awaitable, Callable
import sounddevice as sd
logger = logging.getLogger(__name__)
class PollMonitor:
def __init__(self, configured_device: str | None, interval: float = 2.5):
self.configured_device = configured_device
self.interval = interval
self.on_device_added: Callable[[str], Awaitable[None]] | None = None
self.on_device_removed: Callable[[str], Awaitable[None]] | None = None
self.on_configured_missing: Callable[[], Awaitable[None]] | None = None
self._task: asyncio.Task | None = None
self._known_devices: set[str] = set()
def _current_devices(self) -> set[str]:
try:
return {
dev["name"]
for dev in sd.query_devices()
if dev["max_input_channels"] > 0
}
except Exception:
logger.exception("Fehler beim Abfragen der Audiogeräte")
return self._known_devices.copy()
async def start(self) -> None:
self._known_devices = self._current_devices()
self._task = asyncio.create_task(self._loop())
def stop(self) -> None:
if self._task is not None:
self._task.cancel()
self._task = None
async def _loop(self) -> None:
while True:
await asyncio.sleep(self.interval)
current = self._current_devices()
added = current - self._known_devices
removed = self._known_devices - current
self._known_devices = current
for name in added:
if self.on_device_added:
await self.on_device_added(name)
for name in removed:
if self.on_device_removed:
await self.on_device_removed(name)