feat: implement core/ and platform/ with full test coverage
- platform/interface.py: PlatformClient Protocol + Pydantic models (User, MessageResponse, UserSettings) — no explicit session management, Master handles container lifecycle - platform/mock.py: MockPlatformClient with simulated latency, [MOCK] responses, is_new correctly True only on first creation - core/protocol.py: unified dataclasses for all events and responses (IncomingMessage/Command/Callback, OutgoingMessage/UI/Notification, AuthFlow, ChatContext, SettingsAction, etc.) - core/store.py: StateStore Protocol + InMemoryStore (tests) + SQLiteStore (prod) with JSON serialization - core/chat.py: ChatManager — chat metadata (C1/C2/C3), not container lifecycle (that's the platform's job) - core/auth.py: AuthManager — start_flow / confirm / is_authenticated - core/settings.py: SettingsManager — get/apply with store cache - core/handler.py: EventDispatcher — registry-based routing with keys (command name, action name, attachment type, "*" catch-all) - core/handlers/: register_all() + start/new/message/callback/settings handlers; voice slot falls back to stub text until voice_handler added - conftest.py: sys.path fix so local platform/ shadows stdlib platform - docs/api-contract.md: rewritten for Lambda Lab 3.0 container model 46 tests passing, 0 warnings.
This commit is contained in:
parent
944c383552
commit
36730ae716
27 changed files with 1315 additions and 3 deletions
71
core/handler.py
Normal file
71
core/handler.py
Normal file
|
|
@ -0,0 +1,71 @@
|
|||
# core/handler.py
|
||||
from __future__ import annotations
|
||||
|
||||
from typing import Awaitable, Callable
|
||||
|
||||
import structlog
|
||||
|
||||
from core.auth import AuthManager
|
||||
from core.chat import ChatManager
|
||||
from core.protocol import (
|
||||
IncomingCallback,
|
||||
IncomingCommand,
|
||||
IncomingEvent,
|
||||
IncomingMessage,
|
||||
OutgoingEvent,
|
||||
)
|
||||
from core.settings import SettingsManager
|
||||
from platform.interface import PlatformClient
|
||||
|
||||
logger = structlog.get_logger(__name__)
|
||||
|
||||
HandlerFn = Callable[..., Awaitable[list[OutgoingEvent]]]
|
||||
|
||||
|
||||
class EventDispatcher:
|
||||
def __init__(
|
||||
self,
|
||||
platform: PlatformClient,
|
||||
chat_mgr: ChatManager,
|
||||
auth_mgr: AuthManager,
|
||||
settings_mgr: SettingsManager,
|
||||
) -> None:
|
||||
self._platform = platform
|
||||
self._chat_mgr = chat_mgr
|
||||
self._auth_mgr = auth_mgr
|
||||
self._settings_mgr = settings_mgr
|
||||
self._handlers: dict[type, dict[str, HandlerFn]] = {
|
||||
IncomingCommand: {},
|
||||
IncomingMessage: {},
|
||||
IncomingCallback: {},
|
||||
}
|
||||
|
||||
def register(self, event_type: type, key: str, handler: HandlerFn) -> None:
|
||||
self._handlers[event_type][key] = handler
|
||||
|
||||
async def dispatch(self, event: IncomingEvent) -> list[OutgoingEvent]:
|
||||
event_type = type(event)
|
||||
handlers = self._handlers.get(event_type, {})
|
||||
key = self._routing_key(event)
|
||||
handler = handlers.get(key) or handlers.get("*")
|
||||
|
||||
if handler is None:
|
||||
logger.warning("No handler registered", event_type=event_type.__name__, key=key)
|
||||
return []
|
||||
|
||||
return await handler(
|
||||
event=event,
|
||||
chat_mgr=self._chat_mgr,
|
||||
auth_mgr=self._auth_mgr,
|
||||
settings_mgr=self._settings_mgr,
|
||||
platform=self._platform,
|
||||
)
|
||||
|
||||
def _routing_key(self, event: IncomingEvent) -> str:
|
||||
if isinstance(event, IncomingCommand):
|
||||
return event.command
|
||||
if isinstance(event, IncomingCallback):
|
||||
return event.action
|
||||
if isinstance(event, IncomingMessage) and event.attachments:
|
||||
return event.attachments[0].type
|
||||
return "*"
|
||||
Loading…
Add table
Add a link
Reference in a new issue