Skip to main content
The iii-queue worker decouples producers from consumers: a function publishes a message to a named topic and returns right away, and any function subscribed to that topic processes the message in the background, with retries and a dead-letter queue (DLQ) for messages that keep failing.
iii worker add iii-queue
This page is a quick tour. For retry and back-off settings, queue adapters, and the full function list, see the iii-queue worker docs.

Consuming messages

A function consumes a topic by binding a durable:subscriber trigger to it. The engine runs the function once per message, passing the published data as the payload. Returning normally acknowledges the message; throwing nacks it, so it is retried and eventually dead-lettered.
  1. In a worker, register the consumer function and subscribe it to the topic. If you do not have a worker yet, scaffold one with iii worker init, then edit its source:
iii worker init email-worker --language typescript
import { registerWorker } from "iii-sdk";

const url = process.env.III_URL;
if (!url) throw new Error("III_URL must be set");
const worker = registerWorker(url, { workerName: "email-worker" });

// receives the `data` from each published message
worker.registerFunction("email::send", async (msg: { to: string; subject: string }) => {
  // do the work here; throw to nack and let the message retry
  return { sent: true };
});

worker.registerTrigger({
  type: "durable:subscriber",
  function_id: "email::send",
  config: { topic: "emails" },
});
  1. Add the worker to start it:
iii worker add ./email-worker

Publishing a message

With the consumer running, publish to its topic. The engine delivers the data to every subscriber, so email::send runs once per message:
# publish a message to the "emails" topic
iii trigger iii::durable::publish --json '{"topic":"emails","data":{"to":"a@b.com","subject":"hi"}}'
Open the console and go to the Traces tab to watch the message flow from the publish through to email::send running.

Inspecting Queue Topics

A topic appears here once a function subscribes to it, so these commands inspect the emails topic from above. (Publishing to a topic that nothing subscribes to does not register it, so there is nothing to inspect.) List every topic:
iii trigger engine::queue::list_topics
[{ "name": "emails", "broker_type": "function_queue", "subscriber_count": 1 }]
Get stats for the topic (depth is messages waiting for the consumer, dlq_depth is dead-lettered). A topic whose consumer keeps up sits at depth: 0:
iii trigger engine::queue::topic_stats topic=emails
{ "depth": 0, "consumer_count": 1, "dlq_depth": 0, "config": null }

Inspecting Dead Letter Queue Messages

A message reaches the dead-letter queue only once its subscribed function exhausts its retries, so the DLQ functions return empty until something fails.

Forcing a message into the dead-letter queue

To see the DLQ populated, make email::send fail: change the handler to throw, and set maxRetries: 0 in the trigger’s queue_config so the first failure dead-letters immediately instead of after the default three attempts.
worker.registerFunction("email::send", async () => {
  throw new Error("forced failure");
});

worker.registerTrigger({
  type: "durable:subscriber",
  function_id: "email::send",
  config: { topic: "emails", queue_config: { maxRetries: 0 } },
});
Now publish a message; it fails and lands in the DLQ (exact ids, timestamps, and sizes vary per run):
iii trigger iii::durable::publish --json '{"topic":"emails","data":{"to":"a@b.com","subject":"hi"}}'

Listing topics with dead-lettered messages

iii trigger engine::queue::dlq_topics
[{ "topic": "emails", "broker_type": "function_queue", "message_count": 1 }]

Browsing dead-lettered messages

iii trigger engine::queue::dlq_messages topic=emails
[
  {
    "id": "0b9c…",
    "payload": { "to": "a@b.com", "subject": "hi" },
    "error": "ErrorBody { code: \"invocation_failed\", message: \"forced failure\"...",
    "failed_at": 1718900000,
    "retries": 0,
    "size_bytes": 64
  }
]

Redriving dead-lettered messages

Fix the code back to what it was originally, then move the topic’s dead-lettered messages back to the main queue for reprocessing:
iii trigger iii::queue::redrive topic=emails
{ "queue": "emails", "redriven": 1 }
The fixed function now processes them, so the DLQ is empty again:
iii trigger engine::queue::dlq_messages topic=emails