Agently documentation for building AI applications with stable outputs, observable actions, and durable workflows.
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.
parent.to(prepare).to_sub_flow(child_flow).to(consume)
Without capture or write_back, the bridge does the simplest possible thing:
data.input as its start input.data.input for consume becomes the child’s close snapshot.set_result() or .end(), the parent receives that compat value instead of the snapshot. (See Compatibility.)capture maps parent values into the child’s input and runtime resources:
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:
| Path | Resolves 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 maps the child’s final result back into the parent:
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 value |
Source 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.
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:
prepare_request produces a request context as its return value.to_sub_flow(...) starts the child with that context as the child’s data.input. The parent’s logger resource is forwarded.for_each, drafts each section, summarizes, and writes the result to its own state["report"].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.finalize_request runs in the parent with that data.input.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.
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:
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, runtime_resources={...})
restored.load(saved)
await restored.async_continue_with(root_interrupt_id, {"approved": True})
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.
runtime_resources instead.for_each, if_condition, matchruntime_resources propagate to child via captureresult.<path> falls back to snapshot