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:

  1. 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.
  2. There was an error during the child execution. In this case, the tuple will contain the execution-id and execution-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