Each node specification declaratively defines the role of each node, and these specifications connect to form workflows. Each workflow is defined within a specific namespace, and each runtime environment executes a single namespace. Namespaces are isolated and cannot reference nodes from other namespaces.
+-------------------------------------------------+
| Workflow A |
| +--------------------+ +--------------------+ |
| | Node Specification | | Node Specification | |
| +--------------------+ +--------------------+ |
| \|/ |
| +--------------------+ +--------------------+ |
| | Node Specification | | Node Specification | |
| +--------------------+ +--------------------+ |
+-------------------------------------------------+
| Workflow B |
| +--------------------+ +--------------------+ |
| | Node Specification | | Node Specification | |
| +--------------------+ +--------------------+ |
| \|/ |
| +--------------------+ +--------------------+ |
| | Node Specification | | Node Specification | |
| +--------------------+ +--------------------+ |
+-------------------------------------------------+
The engine does not enforce specific nodes. Nodes connect to the engine through extensions and can be freely added or removed based on service requirements.
To optimize execution, two key processes—compilation and runtime—are employed. These processes reduce complexity and improve performance.
The engine does not expose an API for directly modifying node specifications. Instead, it focuses on loading, compiling, and activating nodes to make them executable.
Users can update node specifications by using a Command-Line Interface (CLI) or
defining workflows that provide an HTTP API for modifications. Typically, these workflows are
defined in the system
namespace.
This approach ensures runtime stability while allowing flexible system expansion.
The loader tracks changes to node specifications and values in real-time through a change stream. When additions, modifications, or deletions occur, the loader dynamically reloads the specifications from the database. These specifications are compiled into executable forms using codecs defined in the schema, with caching and optimization to enhance performance.
Compiled nodes are transformed into symbols and stored in a symbol table. The symbol table connects each symbol's ports based on the port connection information in the node specifications.
+--------------------------+
| Database |
| +--------------------+ |
| | Node Specification | |
| +--------------------+ |
| | Node Specification | |
| +--------------------+ | +-------------------+
| | Node Specification | |-->| Loader |
| +--------------------+ | | +-------------+ |
+--------------------------+ | | Scheme | |
+--------------------------+ | | +-------+ | |
| Database | | | | Codec | | |--+
| +--------+ +--------+ | | | +-------+ | | |
| | Value | | Value | |-->| +-------------+ | |
| +--------+ +--------+ | +-------------------+ |
| +--------+ +--------+ | |
| | Value | | Value | | |
| +--------+ +--------+ | |
+--------------------------+ |
+-------------------------+ |
| Symbol Table | |
| +--------+ +--------+ | |
| | Symbol | | Symbol |<-----------------------------+
| +--------+ +--------+ |
| \|/ |
| +--------+ +--------+ |
| | Symbol | | Symbol | |
| +--------+ +--------+ |
+-------------------------+
Once all nodes in a workflow are loaded into the symbol table and ports are connected, load hooks are executed to activate nodes. If a node is removed, unload hooks deactivate dependent nodes.
Changes in node specifications propagate to all runtime environments.
Activated nodes monitor sockets or files and execute workflows. Each node spawns an independent process, isolating its execution flow from other nodes to optimize resource management.
Nodes open ports through these processes and create writers to send packets to connected nodes. The payload is converted into common types for transmission. Connected nodes create readers to process waiting packets and pass the results to the next node or return them.
Connected nodes monitor whether a new process has opened the port and create readers accordingly. These readers continuously process waiting packets and pass the processed results to the next node or return them.
+-----------------------+ +-----------------------+
| Node A | | Node B |
| +-----------------+ | | +-----------------+ |
| | Port Out |--------------->| Port In | |
| | | | | | | |
| | +-----------+ | | packet | | +-----------+ | |
| | | Process 1 |======================| Process 1 | | |
| | +-----------+ | | | | +-----------+ | |
| | +-----------+ | | packet | | +-----------+ | |
| | | Process 2 |======================| Process 2 | | |
| | +-----------+ | | | | +-----------+ | |
| +-----------------+ | | +-----------------+ |
+-----------------------+ +-----------------------+
Each reader processes packets sequentially and responds. Writers must return packets as responses to ensure smooth node communication and data consistency.
Nodes wait for responses to all sent packets before terminating the process and releasing allocated resources. If an error occurs, the node logs the error and terminates.
Upon process termination, the system releases all associated resources, including open file descriptors, memory, and database transactions. When a parent process terminates, all child processes terminate as well. The parent process typically waits until all child processes have completed.
This architecture ensures efficient node communication, data integrity, and stable execution across workflows.