Architecture#

This page explains how NeMo Flow connects scopes, middleware, plugins, events, subscribers, and exporters.

Architecture Diagram#

This diagram connects the runtime pieces to the layers they inhabit.

        flowchart TB
    subgraph AppLayer[Framework Integrations and Application Code]
        App[Application Code]
        Framework[Framework Integration]
    end

    subgraph BindingLayer[Language Bindings]
        Bindings[Language Bindings]
    end

    subgraph PluginLayer[Plugin and Adaptive Layer]
        PluginSystem[Plugin System]
        Adaptive[Adaptive Component]
    end

    subgraph CoreLayer[Core Runtime]
        Core[Rust Core Runtime]

        subgraph RuntimeState[Runtime State]
            Scope[Scope Stack]
            Registry[Middleware Registries]
        end

        Events[Event Stream]
    end

    subgraph ObsLayer[Subscribers and Observability Backends]
        Subs[Subscribers / Exporters]
        Backends[Files / OTLP / Other Backends]
    end

    App -->|uses| Framework
    App -. direct use .-> Bindings
    App -->|registers and configures| PluginSystem
    Framework -->|calls| Bindings
    Bindings --> Core
    Adaptive -->|activates via| PluginSystem
    PluginSystem -->|installs| Registry
    PluginSystem -->|installs| Subs
    Core -->|updates| Scope
    Core -->|resolves| Registry
    Core -->|emits| Events
    Events -->|fan out to| Subs
    Subs -->|export to| Backends

    class AppLayer grey-hint;
    class BindingLayer grey-hint;
    class PluginLayer grey-hint;
    class CoreLayer grey-hint;
    class ObsLayer grey-hint;
    class RuntimeState grey-lightest;
    class App purple-lightest;
    class Framework yellow-lightest;
    class Bindings green-lightest;
    class PluginSystem green-light;
    class Adaptive blue-lightest;
    class Core green-light;
    class Scope green-light;
    class Registry green-light;
    class Events green-light;
    class Subs green-light;
    class Backends grey-light;
    

Adaptive appears here as a built-in plugin component rather than a separate runtime model because it activates through the same plugin lifecycle.

Runtime Model#

NeMo Flow combines a small number of runtime pieces into one shared execution model:

  • The scope stack answers where work belongs

  • The middleware registries answer what should happen around that work

  • The plugin system installs reusable runtime behavior from configuration

  • The event stream records what happened

  • subscribers consume those events

Every emitted scope, tool, LLM, or mark event attaches to the active scope stack. Every managed tool or LLM call resolves the currently visible middleware before it executes.

Main Runtime Pieces#

These components are the primary building blocks that make up the runtime model.

Scope Stack#

The active scope stack defines the ownership tree for runtime work. It establishes:

  • Parent-child relationships between events

  • Scope-local visibility for middleware and subscribers

  • Cleanup boundaries for scope-owned registrations

  • Isolation across concurrent requests or workers

Middleware Registries#

The middleware registries hold the active intercepts and guardrails for tool and LLM execution. Managed helpers read those registries before invoking the real callback.

Plugin System#

The plugin system installs reusable runtime components from configuration. A plugin can register middleware, subscribers, or related behavior without requiring each application call site to do the work manually.

Event Emission#

The runtime emits structured events for scopes, tools, LLMs, and named marks. Those events are the canonical record of runtime behavior.

Subscribers and Exporters#

Subscribers consume the event stream. Some subscribers stay in-process. Others export that stream into files or tracing systems.

Two Axes of Runtime State#

Runtime state is easiest to understand by separating ownership from process-wide registration.

Scope Ownership#

The scope stack defines:

  • Where work belongs

  • Which scope-local behavior is visible

  • When scope-local registrations are cleaned up

  • Whether concurrent requests stay isolated

Middleware Ownership#

Middleware exists at two levels:

  • global registrations stay active process-wide until removed

  • scope-local registrations are owned by one scope and disappear when that scope closes

That split lets long-lived defaults coexist with request-specific or task-specific behavior.

Managed Execution Pipeline#

Managed tool and LLM execution follows the same high-level order:

  1. Conditional-execution guardrails decide whether work can proceed.

  2. Request intercepts can rewrite the real request.

  3. Sanitize-request guardrails can rewrite the emitted start-event payload.

  4. Execution intercepts wrap or replace the user callback.

  5. The user callback runs.

  6. Sanitize-response guardrails can rewrite the emitted end-event payload.

Two distinctions matter:

  • Intercepts affect the real execution path

  • Sanitize guardrails affect the emitted observability payload

For the expanded request-to-response runtime path, including streaming and subscriber handoff, see Middleware.

Runtime Layers#

From bottom to top, NeMo Flow is organized as:

  1. The Rust core runtime

  2. The plugin and adaptive layer

  3. Language bindings

  4. Framework integrations and application code

  5. Subscribers and observability backends

The details of a binding can vary, but the conceptual model stays the same across those layers.

Design Goal#

NeMo Flow is designed so that application developers, framework integrators, plugin authors, and observability consumers all reason about the same runtime semantics. One conceptual model should remain stable even when the binding or integration style changes.