Documentation Index
Fetch the complete documentation index at: https://motiadev-feat-improve-erros-if-trigger-does-not-exists.mintlify.app/llms.txt
Use this file to discover all available pages before exploring further.
What “writing a trigger” means
Workers don’t only call functions; they can also source events. When a worker advertises a
trigger type (e.g. webhook, a custom schedule, or any external event source you implement), any
other worker can bind its functions to that type. The engine routes binding requests to your
trigger type’s handler, and your worker invokes the bound functions when the underlying event
fires.
This page is about authoring new trigger types from inside your worker. For invoking functions and
binding triggers to functions from the consumer side, see Using iii / Triggers.
Declare a trigger type
Inside the worker, register the trigger type once during startup. You pass an id and
description plus a handler with registerTrigger / unregisterTrigger callbacks. The engine
calls these whenever a consumer binds or unbinds a function against your trigger type, so your
worker can keep its own table of { trigger id → function id, config } bindings.
registerTriggerType returns a typed handle. Use it later to bind functions you own to this
trigger type, or to tear the type down.
Node / TypeScript
Python
Rust
import { registerWorker } from "iii-sdk";
import type { TriggerConfig, TriggerHandler } from "iii-sdk";
const worker = registerWorker(process.env.III_URL);
type WebhookTriggerConfig = {
path: string;
secret?: string;
};
const webhookHandler: TriggerHandler<WebhookTriggerConfig> = {
async registerTrigger(config: TriggerConfig<WebhookTriggerConfig>) {
// Stash { config.id, config.function_id, config.config } so the
// worker's HTTP listener knows which function to invoke when a
// request matches `config.config.path`.
},
async unregisterTrigger(config: TriggerConfig<WebhookTriggerConfig>) {
// Remove the binding from the worker's table.
},
};
const webhook = worker.registerTriggerType(
{ id: "webhook", description: "Incoming webhook trigger" },
webhookHandler,
);
import os
from iii import (
InitOptions,
RegisterTriggerTypeInput,
TriggerConfig,
TriggerHandler,
register_worker,
)
worker = register_worker(
os.environ.get("III_URL"),
InitOptions(worker_name="webhook-worker"),
)
class WebhookHandler(TriggerHandler):
async def register_trigger(self, config: TriggerConfig) -> None:
# Stash config.id, config.function_id, config.config
...
async def unregister_trigger(self, config: TriggerConfig) -> None:
...
webhook = worker.register_trigger_type(
RegisterTriggerTypeInput(id="webhook", description="Incoming webhook trigger"),
WebhookHandler(),
)
use iii_sdk::{
InitOptions, RegisterTriggerType, TriggerConfig, TriggerHandler, register_worker,
};
let url = std::env::var("III_URL").expect("III_URL must be set");
let worker = register_worker(&url, InitOptions::default());
struct WebhookHandler;
#[async_trait::async_trait]
impl TriggerHandler for WebhookHandler {
async fn register_trigger(&self, config: TriggerConfig) -> Result<(), iii_sdk::IIIError> {
// Stash config.id, config.function_id, config.config
Ok(())
}
async fn unregister_trigger(&self, config: TriggerConfig) -> Result<(), iii_sdk::IIIError> {
Ok(())
}
}
let webhook = worker.register_trigger_type(
RegisterTriggerType::new("webhook", "Incoming webhook trigger", WebhookHandler),
);
For typed config and call-payload schemas, attach Pydantic, Zod, or schemars::JsonSchema types
to the registration in your language of choice. See each SDK’s reference for the exact builder.
Dispatch events to bound functions
There is no special “fire” API. When the underlying event source delivers something (an incoming
HTTP request, a cron tick, a webhook hit), your worker looks up the bindings it stashed in its
registerTrigger callback and invokes each bound function via worker.trigger(...).
Node / TypeScript
Python
Rust
// Inside the worker's HTTP listener, after matching a request to a binding:
await worker.trigger({
function_id: binding.function_id,
payload: { method, headers, body },
});
# Inside the worker's HTTP listener, after matching a request to a binding:
worker.trigger({
"function_id": binding["function_id"],
"payload": {"method": method, "headers": headers, "body": body},
})
use iii_sdk::TriggerRequest;
use serde_json::json;
// Inside the worker's HTTP listener, after matching a request to a binding:
worker
.trigger(TriggerRequest {
function_id: binding.function_id.clone(),
payload: json!({ "method": method, "headers": headers, "body": body }),
action: None,
timeout_ms: None,
})
.await?;
On each dispatched event, the engine evaluates the consumer’s config and optional
condition_function_id, then routes matching invocations to the bound function and returns the
result to the caller.
Unregister a trigger type
registerTriggerType returns a handle with an unregister() method that tears down the trigger
type at runtime. When the worker disconnects, all trigger types it advertised are removed
automatically and the engine stops routing events that depended on them.
Node / TypeScript
Python
Rust
worker.unregister_trigger_type(
RegisterTriggerTypeInput(id="webhook", description="Incoming webhook trigger"),
)
worker.unregister_trigger_type("webhook");
What goes in Worker Docs
The trigger type id, the shape of the config consumers pass when binding, the call-payload
shape your worker delivers to bound functions, the event ordering guarantees, and any
back-pressure or retry semantics belong in this worker’s Worker Docs so consumers know what to
pass when they call registerTrigger. Keep iii-level concepts (the trigger binding model itself,
condition gates) here; document the per-type specifics there.