Skip to content

Sub-Flow

Languages: English · 中文

to_sub_flow(child_flow, ...) lets a parent flow embed a child flow as a single chunk. The child runs to its own close, then the parent continues with whatever the child produced.

Plain composition

python
parent.to(prepare).to_sub_flow(child_flow).to(consume)

Without capture or write_back, the bridge does the simplest possible thing:

  • The child receives the parent's current data.input as its start input.
  • After the child closes, the parent's data.input for consume becomes the child's close snapshot.
  • If the child wrote a compatibility result via the deprecated set_result() or .end(), the parent receives that compat value instead of the snapshot. (See Compatibility.)

capture — selecting parent → child input

capture maps parent values into the child's input and runtime resources:

python
parent.to(prepare_request).to_sub_flow(
    child_flow,
    capture={
        "input": "value",                       # child's start input = parent's current data.input
        "resources": {"logger": "resources.logger"},
    },
)

Common capture paths:

PathResolves to
"value"the parent's current data.input
"state.<key>"a value from parent's state
"resources.<name>"a parent runtime resource

The right column is mapped onto the child's input or resources by the keys on the left.

write_back — child result → parent

write_back maps the child's final result back into the parent:

python
parent.to(prepare).to_sub_flow(
    child_flow,
    capture={"input": "value"},
    write_back={"value": "result.report"},
).to(finalize)

Resolution rules for write_back:

write_back valueSource preference
"result"child compat result if present, otherwise child close snapshot
"result.<path>"first try the same path inside child compat result; fall back to the matching path inside the close snapshot
"snapshot"the close snapshot directly (skip compat result)
"snapshot.<path>"path inside the snapshot

The value key on the left side puts the resolved value back into the parent's data.input for the next chunk. Other keys (state.<name>) write into parent state.

This is why the same result.<path> syntax works for both legacy compat-result-style children and new state-first children — the lookup tries compat first, then falls back to the snapshot.

Worked example

python
def build_child_flow():
    child = TriggerFlow(name="child")
    (
        child.if_condition(has_multiple_sections)
            .to(use_multi_section_mode)
        .else_condition()
            .to(use_single_section_mode)
        .end_condition()
        .to(list_sections)
        .for_each()
            .to(draft_section)
        .end_for_each()
        .to(summarize_child_report)
    )
    return child


def build_parent_flow():
    parent = TriggerFlow(name="parent")
    parent.update_runtime_resources(logger=SimpleLogger())
    parent.to(prepare_request).to_sub_flow(
        build_child_flow(),
        capture={
            "input": "value",
            "resources": {"logger": "resources.logger"},
        },
        write_back={
            "value": "result.report",
        },
    ).to(finalize_request)
    return parent

What happens:

  1. prepare_request produces a request context as its return value.
  2. to_sub_flow(...) starts the child with that context as the child's data.input. The parent's logger resource is forwarded.
  3. The child branches, fans out via for_each, drafts each section, summarizes, and writes the result to its own state["report"].
  4. The bridge resolves write_back={"value": "result.report"}: it looks for report first in any compat result the child set, then in the child's close snapshot, finds it, and assigns it as the parent's next data.input.
  5. finalize_request runs in the parent with that data.input.

Stream items cross sub-flow boundaries

Items pushed via data.async_put_into_stream(...) inside the child show up in the parent execution's runtime stream. From an external consumer's point of view, sub-flows look like part of the same execution.

Child pauses project to the parent

If a child flow calls pause_for(...), the parent execution becomes waiting too. External systems still manage only the parent execution id and the parent interrupt id:

python
execution = parent_flow.create_execution(auto_close=False)
await execution.async_start(input_value)

root_interrupt_id = next(iter(execution.get_pending_interrupts()))
saved = execution.save()

restored = parent_flow.create_execution(auto_close=False)
await restored.async_load(saved, runtime_resources={...})
await restored.async_continue_with(
    root_interrupt_id,
    {"approved": True},
    resume_request_id="approval-webhook-42",
)

The projected interrupt includes sub_flow_frame_id and local_interrupt_id for debugging, but callers should treat the parent interrupt id as the public handle. After the child finishes, write_back runs normally and the parent continues downstream.

For a prearranged document-review approval gate, see examples/step_by_step/11-triggerflow-20_document_review_subflow_pause_resume.py. The child sub-flow has an explicit pause chunk and waits through when("LegalApprovalSubmitted"); the parent still saves, reloads, and resumes through the projected root interrupt.

When to use a sub-flow

  • The child is reusable — used by multiple parent flows or independently.
  • The child has its own well-defined contract (input + result) you'd want to test in isolation.
  • You want to keep the parent flow shorter and more readable.

When not to use a sub-flow

  • The child is only one or two chunks. Inline them.
  • You're using sub-flow purely as a way to share state. Use a parent function or runtime_resources instead.
  • You'd be tempted to share runtime stream filtering between parent and child — separate concerns, don't entangle.

See also