mirror of
https://github.com/bytedance/deer-flow.git
synced 2026-05-02 23:08:22 +00:00
- Replace 6 identical getter functions with _require() factory. - Inline 3 _make_*_repo() factories into langgraph_runtime(), call get_session_factory() once instead of 3 times. - Add thread_meta upsert in start_run (services.py). Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
119 lines
4.4 KiB
Python
119 lines
4.4 KiB
Python
"""Centralized accessors for singleton objects stored on ``app.state``.
|
|
|
|
**Getters** (used by routers): raise 503 when a required dependency is
|
|
missing, except ``get_store`` and ``get_thread_meta_repo`` which return
|
|
``None``.
|
|
|
|
Initialization is handled directly in ``app.py`` via :class:`AsyncExitStack`.
|
|
"""
|
|
|
|
from __future__ import annotations
|
|
|
|
from collections.abc import AsyncGenerator
|
|
from contextlib import AsyncExitStack, asynccontextmanager
|
|
|
|
from fastapi import FastAPI, HTTPException, Request
|
|
|
|
from deerflow.runtime import RunManager
|
|
|
|
|
|
@asynccontextmanager
|
|
async def langgraph_runtime(app: FastAPI) -> AsyncGenerator[None, None]:
|
|
"""Bootstrap and tear down all LangGraph runtime singletons.
|
|
|
|
Usage in ``app.py``::
|
|
|
|
async with langgraph_runtime(app):
|
|
yield
|
|
"""
|
|
from deerflow.agents.checkpointer.async_provider import make_checkpointer
|
|
from deerflow.config import get_app_config
|
|
from deerflow.persistence.engine import close_engine, get_session_factory, init_engine_from_config
|
|
from deerflow.runtime import make_store, make_stream_bridge
|
|
from deerflow.runtime.events.store import make_run_event_store
|
|
|
|
async with AsyncExitStack() as stack:
|
|
app.state.stream_bridge = await stack.enter_async_context(make_stream_bridge())
|
|
|
|
# Initialize persistence engine BEFORE checkpointer so that
|
|
# auto-create-database logic runs first (postgres backend).
|
|
config = get_app_config()
|
|
await init_engine_from_config(config.database)
|
|
|
|
app.state.checkpointer = await stack.enter_async_context(make_checkpointer())
|
|
app.state.store = await stack.enter_async_context(make_store())
|
|
|
|
# Initialize repositories — one get_session_factory() call for all.
|
|
sf = get_session_factory()
|
|
if sf is not None:
|
|
from deerflow.persistence.repositories.feedback_repo import FeedbackRepository
|
|
from deerflow.persistence.repositories.run_repo import RunRepository
|
|
from deerflow.persistence.repositories.thread_meta_repo import ThreadMetaRepository
|
|
|
|
app.state.run_store = RunRepository(sf)
|
|
app.state.feedback_repo = FeedbackRepository(sf)
|
|
app.state.thread_meta_repo = ThreadMetaRepository(sf)
|
|
else:
|
|
from deerflow.runtime.runs.store.memory import MemoryRunStore
|
|
|
|
app.state.run_store = MemoryRunStore()
|
|
app.state.feedback_repo = None
|
|
app.state.thread_meta_repo = None
|
|
|
|
# Run event store (has its own factory with config-driven backend selection)
|
|
run_events_config = getattr(config, "run_events", None)
|
|
app.state.run_event_store = make_run_event_store(run_events_config)
|
|
|
|
# RunManager with store backing for persistence
|
|
app.state.run_manager = RunManager(store=app.state.run_store)
|
|
|
|
try:
|
|
yield
|
|
finally:
|
|
await close_engine()
|
|
|
|
|
|
# ---------------------------------------------------------------------------
|
|
# Getters -- called by routers per-request
|
|
# ---------------------------------------------------------------------------
|
|
|
|
|
|
def _require(attr: str, label: str):
|
|
"""Create a FastAPI dependency that returns ``app.state.<attr>`` or 503."""
|
|
|
|
def dep(request: Request):
|
|
val = getattr(request.app.state, attr, None)
|
|
if val is None:
|
|
raise HTTPException(status_code=503, detail=f"{label} not available")
|
|
return val
|
|
|
|
dep.__name__ = dep.__qualname__ = f"get_{attr}"
|
|
return dep
|
|
|
|
|
|
get_stream_bridge = _require("stream_bridge", "Stream bridge")
|
|
get_run_manager = _require("run_manager", "Run manager")
|
|
get_checkpointer = _require("checkpointer", "Checkpointer")
|
|
get_run_event_store = _require("run_event_store", "Run event store")
|
|
get_feedback_repo = _require("feedback_repo", "Feedback")
|
|
get_run_store = _require("run_store", "Run store")
|
|
|
|
|
|
def get_store(request: Request):
|
|
"""Return the global store (may be ``None`` if not configured)."""
|
|
return getattr(request.app.state, "store", None)
|
|
|
|
|
|
def get_thread_meta_repo(request: Request):
|
|
"""Return the ThreadMetaRepository, or None if not available."""
|
|
return getattr(request.app.state, "thread_meta_repo", None)
|
|
|
|
|
|
async def get_current_user(request: Request) -> str | None:
|
|
"""Extract user identity from request.
|
|
|
|
Phase 2: always returns None (no authentication).
|
|
Phase 3: extract user_id from JWT / session / API key header.
|
|
"""
|
|
return None
|