Extension Functions
Obelisk automatically generates extension functions for each workflow and activity function defined in your WIT (WebAssembly Interface Type) files. These extensions simplify submitting, awaiting, and scheduling child executions from a parent workflow. They streamline common tasks associated with workflow execution and are a key part of Obelisk's structured concurrency model.
Understanding Extension Functions
Consider an activity function defined in WIT like this:
package stargazers:llm;
interface llm {
respond: func(user-prompt: string, settings-json: string) -> result<string, string>;
}
Obelisk will auto-generate the following extension functions in a separate WIT package for each exported function:
// Generated by Obelisk.
package stargazers:llm-obelisk-ext;
interface llm {
use obelisk:types/execution@1.0.0.{execution-id, join-set-id};
use obelisk:types/time@1.0.0.{schedule-at};
use obelisk:types/execution@1.0.0.{execution-error};
respond-submit: func(join-set-id: borrow<join-set-id>, user-prompt: string, settings-json: string) -> execution-id;
respond-await-next: func(join-set-id: borrow<join-set-id>) -> result<tuple<execution-id, result<string, string>>, tuple<execution-id, execution-error>>;
respond-schedule: func(schedule-at: schedule-at, user-prompt: string, settings-json: string) -> execution-id;
}
Let's break down each generated function:
Submitting child executions with -submit
This function is used to submit a request for the original function to be executed. It takes the join-set-id
and the original parameters as input. It returns an execution-id
, which uniquely identifies this particular execution request. The join-set-id
associates this execution with a specific join set
, allowing you to manage multiple concurrent executions. This function does not wait for the execution to complete; it simply submits the request.
Awaiting next result with -await-next
This function is used to wait for the next result from a given join-set-id
. It takes the join-set-id
as input and returns a result
. The result contains a tuple with two possible outcomes which must be handled:
- The workflow function has finished successfully. In this case, the tuple will hold the
execution-id
that identifies the finished execution and the original return value. - There was an error during the child execution. In this case, the tuple will contain the
execution-id
andexecution-error
as defined in obelisk:types
By repeatedly calling respond-await-next
after submitting multiple executions on a join set, you can retrieve results as they become available, in the order they complete.
Scheduling a new execution with -schedule
This function allows you to schedule the execution of the original function at a specific time in the future. It takes a schedule-at
(defined in obelisk:types/time) and the original parameters as input. Like respond-submit
, it returns an execution-id
. The Obelisk runtime will ensure the function is executed at (or as close as possible to) the specified time. It's important to note that scheduled executions do not use join sets. This is because scheduling inherently breaks the parent-child relationship that join sets rely on for structured concurrency. Consequently, a scheduled function's result cannot be directly awaited by the caller. Scheduled functions are "fire-and-forget" from the perspective of the scheduling workflow. If you need to retrieve the result, you would typically use the persistent sleep
function instead.
Key Concepts and Benefits
-
Asynchronous Execution:
submit
andawait-next
enable asynchronous execution. You can submit multiple requests without blocking, and then process results as they become available. -
Concurrency Management: Join sets , combined with
submit
andawait-next
, provide a powerful mechanism for structured concurrency. You can group related executions together and retrieve their results efficiently. -
Scheduled Execution: The
schedule
function allows for time-based execution, enabling use cases like delayed tasks and periodically scheduled jobs. -
Type Safety: The generated WIT file ensures type safety. The types used in the extension functions (
execution-id
,join-set-id
,schedule-at
,execution-error
) are defined in the ]obelisk:types WIT files, providing a clear and consistent interface. -
Simplified Workflow Interaction: Obelisk workflows execute in a single-threaded, synchronous manner within their WebAssembly sandbox. This eliminates the complexities often associated with asynchronous programming, such as callbacks, async/await keywords, and the need to manage concurrent threads directly within the workflow code. You write your workflow logic as if it were a simple, sequential program, even when it involves multiple activity calls or sub-workflows. Obelisk handles the underlying asynchronous operations and concurrency for you.