refactor(graph): Use piped Runnables for worker nodes to enable subgraph event streaming
This commit is contained in:
@@ -16,17 +16,9 @@ from ea_chatbot.graph.nodes.summarize_conversation import summarize_conversation
|
|||||||
_DATA_ANALYST_WORKER = create_data_analyst_worker()
|
_DATA_ANALYST_WORKER = create_data_analyst_worker()
|
||||||
_RESEARCHER_WORKER = create_researcher_worker()
|
_RESEARCHER_WORKER = create_researcher_worker()
|
||||||
|
|
||||||
def data_analyst_worker_node(state: AgentState) -> dict:
|
# Define worker nodes as piped runnables to enable subgraph event propagation
|
||||||
"""Wrapper node for the Data Analyst subgraph with state mapping."""
|
data_analyst_worker_runnable = prepare_worker_input | _DATA_ANALYST_WORKER | merge_worker_output
|
||||||
worker_input = prepare_worker_input(state)
|
researcher_worker_runnable = prepare_researcher_input | _RESEARCHER_WORKER | merge_researcher_output
|
||||||
worker_result = _DATA_ANALYST_WORKER.invoke(worker_input)
|
|
||||||
return merge_worker_output(worker_result)
|
|
||||||
|
|
||||||
def researcher_worker_node(state: AgentState) -> dict:
|
|
||||||
"""Wrapper node for the Researcher subgraph with state mapping."""
|
|
||||||
worker_input = prepare_researcher_input(state)
|
|
||||||
worker_result = _RESEARCHER_WORKER.invoke(worker_input)
|
|
||||||
return merge_researcher_output(worker_result)
|
|
||||||
|
|
||||||
def main_router(state: AgentState) -> str:
|
def main_router(state: AgentState) -> str:
|
||||||
"""Route from query analyzer based on initial assessment."""
|
"""Route from query analyzer based on initial assessment."""
|
||||||
@@ -52,8 +44,8 @@ def create_workflow(
|
|||||||
query_analyzer=query_analyzer_node,
|
query_analyzer=query_analyzer_node,
|
||||||
planner=planner_node,
|
planner=planner_node,
|
||||||
delegate=delegate_node,
|
delegate=delegate_node,
|
||||||
data_analyst_worker=data_analyst_worker_node,
|
data_analyst_worker=data_analyst_worker_runnable,
|
||||||
researcher_worker=researcher_worker_node,
|
researcher_worker=researcher_worker_runnable,
|
||||||
reflector=reflector_node,
|
reflector=reflector_node,
|
||||||
synthesizer=synthesizer_node,
|
synthesizer=synthesizer_node,
|
||||||
clarification=clarification_node,
|
clarification=clarification_node,
|
||||||
|
|||||||
@@ -19,14 +19,13 @@ def auth_header(mock_user):
|
|||||||
yield {"Authorization": f"Bearer {token}"}
|
yield {"Authorization": f"Bearer {token}"}
|
||||||
app.dependency_overrides.clear()
|
app.dependency_overrides.clear()
|
||||||
|
|
||||||
def test_persistence_integration_success(auth_header, mock_user):
|
def test_persistence_integration_success(auth_header, mock_user):
|
||||||
"""Test that messages and plots are persisted correctly during streaming."""
|
"""Test that messages and plots are persisted correctly during streaming."""
|
||||||
mock_events = [
|
mock_events = [
|
||||||
{"event": "on_chat_model_stream", "name": "summarizer", "data": {"chunk": "Final answer"}},
|
{"event": "on_chat_model_stream", "metadata": {"langgraph_node": "synthesizer"}, "data": {"chunk": "Final answer"}},
|
||||||
{"event": "on_chain_end", "name": "summarizer", "data": {"output": {"messages": [{"content": "Final answer"}]}}},
|
{"event": "on_chain_end", "name": "synthesizer", "data": {"output": {"messages": [{"content": "Final answer"}]}}},
|
||||||
{"event": "on_chain_end", "name": "summarize_conversation", "data": {"output": {"summary": "New summary"}}}
|
{"event": "on_chain_end", "name": "summarize_conversation", "data": {"output": {"summary": "New summary"}}}
|
||||||
]
|
]
|
||||||
|
|
||||||
async def mock_astream_events(*args, **kwargs):
|
async def mock_astream_events(*args, **kwargs):
|
||||||
for event in mock_events:
|
for event in mock_events:
|
||||||
yield event
|
yield event
|
||||||
|
|||||||
@@ -1,11 +1,11 @@
|
|||||||
import pytest
|
import pytest
|
||||||
from ea_chatbot.graph.workflow import create_workflow, data_analyst_worker_node
|
from ea_chatbot.graph.workflow import create_workflow, data_analyst_worker_runnable
|
||||||
from ea_chatbot.graph.state import AgentState
|
from ea_chatbot.graph.state import AgentState
|
||||||
from unittest.mock import MagicMock, patch
|
from unittest.mock import MagicMock, patch
|
||||||
from langchain_core.messages import HumanMessage, AIMessage
|
from langchain_core.messages import HumanMessage, AIMessage
|
||||||
|
|
||||||
def test_worker_merge_sets_summary_for_reflector():
|
def test_worker_merge_sets_summary_for_reflector(monkeypatch):
|
||||||
"""Verify that worker node (wrapper) sets the 'summary' field for the Reflector."""
|
"""Verify that worker node (runnable) sets the 'summary' field for the Reflector."""
|
||||||
|
|
||||||
state = AgentState(
|
state = AgentState(
|
||||||
messages=[HumanMessage(content="test")],
|
messages=[HumanMessage(content="test")],
|
||||||
@@ -21,17 +21,23 @@ def test_worker_merge_sets_summary_for_reflector():
|
|||||||
summary="Initial Planner Summary" # Stale summary
|
summary="Initial Planner Summary" # Stale summary
|
||||||
)
|
)
|
||||||
|
|
||||||
# Mock the compiled worker subgraph to return a specific result
|
# Create a mock for the invoke method
|
||||||
with patch("ea_chatbot.graph.workflow._DATA_ANALYST_WORKER") as mock_worker:
|
mock_invoke = MagicMock()
|
||||||
mock_worker.invoke.return_value = {
|
mock_invoke.return_value = {
|
||||||
"result": "Actual Worker Findings",
|
"summary": "Actual Worker Findings",
|
||||||
"messages": [AIMessage(content="Internal")],
|
"messages": [AIMessage(content="Actual Worker Findings")],
|
||||||
"vfs_state": {},
|
"vfs": {},
|
||||||
"plots": []
|
"plots": []
|
||||||
}
|
}
|
||||||
|
|
||||||
# Execute the wrapper node
|
# Manually replace the runnable with a mock object that has an invoke method
|
||||||
updates = data_analyst_worker_node(state)
|
mock_runnable = MagicMock()
|
||||||
|
mock_runnable.invoke = mock_invoke
|
||||||
|
monkeypatch.setattr("ea_chatbot.graph.workflow.data_analyst_worker_runnable", mock_runnable)
|
||||||
|
|
||||||
|
# Execute via the module reference (which is now mocked)
|
||||||
|
from ea_chatbot.graph.workflow import data_analyst_worker_runnable
|
||||||
|
updates = data_analyst_worker_runnable.invoke(state)
|
||||||
|
|
||||||
# Verify that 'summary' is in updates and has the worker result
|
# Verify that 'summary' is in updates and has the worker result
|
||||||
assert "summary" in updates
|
assert "summary" in updates
|
||||||
|
|||||||
Reference in New Issue
Block a user