2026-03-21 22:13:41 -07:00
|
|
|
from langgraph.graph import StateGraph
|
|
|
|
|
|
|
|
|
|
from .configuration import Configuration
|
|
|
|
|
from .nodes import (
|
2026-03-21 23:02:09 -07:00
|
|
|
assign_slide_themes,
|
2026-03-21 22:13:41 -07:00
|
|
|
create_presentation_slides,
|
|
|
|
|
create_slide_audio,
|
|
|
|
|
generate_slide_scene_codes,
|
|
|
|
|
)
|
|
|
|
|
from .state import State
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
def build_graph():
|
|
|
|
|
workflow = StateGraph(State, config_schema=Configuration)
|
|
|
|
|
|
|
|
|
|
workflow.add_node("create_presentation_slides", create_presentation_slides)
|
|
|
|
|
workflow.add_node("create_slide_audio", create_slide_audio)
|
2026-03-21 23:02:09 -07:00
|
|
|
workflow.add_node("assign_slide_themes", assign_slide_themes)
|
2026-03-21 22:13:41 -07:00
|
|
|
workflow.add_node("generate_slide_scene_codes", generate_slide_scene_codes)
|
|
|
|
|
|
2026-03-21 23:02:09 -07:00
|
|
|
# Fan-out: after slides are parsed, run audio generation and theme
|
|
|
|
|
# assignment in parallel (themes only need slide metadata, not audio).
|
2026-03-21 22:13:41 -07:00
|
|
|
workflow.add_edge("__start__", "create_presentation_slides")
|
|
|
|
|
workflow.add_edge("create_presentation_slides", "create_slide_audio")
|
2026-03-21 23:02:09 -07:00
|
|
|
workflow.add_edge("create_presentation_slides", "assign_slide_themes")
|
|
|
|
|
|
|
|
|
|
# Fan-in: scene code generation waits for both audio and themes.
|
2026-03-21 22:13:41 -07:00
|
|
|
workflow.add_edge("create_slide_audio", "generate_slide_scene_codes")
|
2026-03-21 23:02:09 -07:00
|
|
|
workflow.add_edge("assign_slide_themes", "generate_slide_scene_codes")
|
|
|
|
|
|
2026-03-21 22:13:41 -07:00
|
|
|
workflow.add_edge("generate_slide_scene_codes", "__end__")
|
|
|
|
|
|
|
|
|
|
graph = workflow.compile()
|
|
|
|
|
graph.name = "Surfsense Video Presentation"
|
|
|
|
|
|
|
|
|
|
return graph
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
graph = build_graph()
|