PSA/ee/docs/extension-system/runner_guest_abi.md
Hermes 284313f908
Some checks are pending
Bidi Control Character Guard / bidi-control-guard (push) Waiting to run
Circular Dependency Check / Check for new circular dependencies (push) Waiting to run
Citus Migration Smoke / Combined migrations on single-node Citus (push) Waiting to run
E2E Fresh Install Tests / fresh-install-e2e (push) Waiting to run
ext-v2 guardrails / Run ext-v2 guard and ESLint (push) Waiting to run
Integration Tests / Check for relevant changes (push) Waiting to run
Integration Tests / ${{ (github.event_name == 'schedule' || github.event.inputs.suite == 'full') && 'Full integration suite' || 'Tier-1 integration subset' }} (push) Blocked by required conditions
Mobile checks / Mobile lint + typecheck (push) Waiting to run
Mobile checks / Mobile unit tests (push) Waiting to run
Mobile checks / Mobile dependency audit (report) (push) Waiting to run
Mobile checks / Mobile reproducibility checks (push) Waiting to run
Secrets guard (env backups) / Ensure no tracked env backup files (push) Waiting to run
Temporal Readiness / fast-readiness (push) Waiting to run
Temporal Readiness / docker-parity (push) Waiting to run
TypeScript Type Check / Nx affected typecheck (push) Waiting to run
Unit Tests / Skipped-test budget (push) Waiting to run
Unit Tests / Nx affected unit tests (push) Waiting to run
Unit Tests / Server unit coverage (informational) (push) Waiting to run
Validate Tenant Management Schema / Check for relevant changes (push) Waiting to run
Validate Tenant Management Schema / Validate Tenant Management Schema (push) Blocked by required conditions
EE Workflows Build Guard / ee-workflows-build-guard (push) Waiting to run
Initial import of AlgaPSA codebase from PSA server
Excluded: .git, node_modules, secrets/, compose.env, assemblyscript tgz

Source: /opt/alga-psa on psa.joliet.tech
2026-06-22 16:12:17 -05:00

124 lines
5.7 KiB
Markdown

# Runner Guest ABI (EE)
This document specifies the ABI between the Runner (host) and Extension bundles (guest WebAssembly module).
Scope
- Execution: how the host calls your handler and how you return a response
- Memory/alloc conventions
- Host imports provided (logging, HTTP egress)
- Data formats and constraints
Overview
- The guest exports a linear memory and functions: alloc, handler, and optionally dealloc
- The host writes a normalized request JSON into guest memory and calls handler
- The guest writes a response JSON into guest memory and indicates its location via an out-pointer tuple
Guest exports
- memory: (export "memory") (required)
- A standard linear memory the host can read/write
- alloc: (export "alloc") (required)
- Signature: alloc(size: i32) -> i32
- Returns a pointer to a region of guest memory at least size bytes
- dealloc: (export "dealloc") (optional)
- Signature: dealloc(ptr: i32, size: i32) -> void
- If exported, the host may call dealloc for input, output, and tuple buffers after use
- handler: (export "handler") (required)
- Signature: handler(req_ptr: i32, req_len: i32, out_ptr: i32) -> i32
- Arguments:
- req_ptr/req_len: point to a UTF-8 JSON request payload provided by host (see Request JSON)
- out_ptr: points to an 8-byte area in guest memory where the guest must write back a pair of little-endian i32 values: (resp_ptr, resp_len)
- resp_ptr: pointer to the start of the response JSON bytes in guest memory
- resp_len: length in bytes of the response JSON
- Return value: 0 for success; non-zero indicates application-level error. Traps terminate execution.
Request JSON (host -> guest)
- Bytes at (req_ptr, req_len) are UTF-8 JSON with the shape:
{
"context": {
"request_id": string | null,
"tenant_id": string,
"extension_id": string,
"version_id": string | null
},
"http": {
"method": string, // "GET", "POST", ...
"path": string, // "/route"
"query": { [k: string]: string },
"headers": { [k: string]: string },
"body_b64": string | null // base64-encoded body (if any)
}
}
Response JSON (guest -> host)
- Guest allocates a buffer via alloc(len), writes UTF-8 JSON, then writes the tuple (resp_ptr, resp_len) at out_ptr
- JSON shape expected by host:
{
"status": number, // e.g., 200
"headers": { [k: string]: string }, // optional
"body_b64": string | null // base64-encoded body (optional)
}
- If the guest returns non-JSON bytes, the host will treat them as an opaque payload and base64-encode them with status 200 and no headers
Host imports (module "alga")
- alga.log_info(ptr: i32, len: i32) -> void
- Reads UTF-8 string from guest memory and logs at info level
- alga.log_error(ptr: i32, len: i32) -> void
- Reads UTF-8 string from guest memory and logs at error level
- alga.http.fetch(req_ptr: i32, req_len: i32, out_ptr: i32) -> i32
- Asynchronous host function exposed synchronously in the ABI; returns 0 on success, non-zero on error (trap on severe errors)
- Request JSON (at req_ptr/req_len):
{
"url": string, // required
"method": string, // optional, default "GET"
"headers": { [k: string]: string }, // optional
"body_b64": string | null // optional
}
- Allowlist enforcement: the host validates URL host against EXT_EGRESS_ALLOWLIST (comma-separated hostnames). Exact or subdomain match required. If denied, returns error.
- On success, host writes response JSON to guest memory (alloc used) and stores (resp_ptr, resp_len) at out_ptr with shape:
{
"status": number,
"headers": { [k: string]: string },
"body_b64": string
}
- Notes: Size/time limits may be enforced by the host; avoid large payloads
Limits & timeouts
- Per-invocation limits may be applied by the host:
- Timeout: context.limits.timeout_ms (default configured by host); exceeding deadline interrupts execution
- Memory: context.limits.memory_mb (default configured by host); allocations beyond limit will fail
- Egress allowlist: EXT_EGRESS_ALLOWLIST environment variable on the Runner controls which hosts alga.http.fetch can access
- Request/response sizes: the host may enforce maximum sizes for inbound/outbound bodies; exceeding limits will error
Guest design guidance
- Keep handler stateless and idempotent where practical; rely on host-brokered I/O only
- Always check for missing/optional fields in request JSON
- Return normalized response JSON whenever possible (status, headers, body_b64)
- Use alga.http.fetch for outbound HTTP only to allowed domains
- Free memory when exporting dealloc; host will attempt to call it if present
Minimal pseudo-code
- Pseudocode guest outline:
export function alloc(sz: i32): i32 { /* ... */ }
export function dealloc(ptr: i32, sz: i32): void { /* ... */ }
export function handler(req_ptr: i32, req_len: i32, out_ptr: i32): i32 {
const req = JSON.parse(loadString(req_ptr, req_len));
const res = { status: 200, headers: { "content-type": "application/json" }, body_b64: b64encode(utf8("{\"ok\":true}")) };
const buf = utf8(JSON.stringify(res));
const resp_ptr = alloc(buf.length);
storeBytes(resp_ptr, buf);
storeI32(out_ptr + 0, resp_ptr);
storeI32(out_ptr + 4, buf.length);
return 0;
}
Error handling
- handler returns non-zero to signal application errors; the host maps this to a 500 execute_failed with the code
- Traps (e.g., out-of-bounds, allowlist denial, host errors) abort execution; the host returns 500 with an error message
Versioning
- This is the initial MVP ABI for the Runner (EE). Future revisions may add:
- Structured error mapping
- Built-in size/time limit introspection
- Additional host imports (KV/doc storage, secrets)