by inngest
Universally Triggered Agent Harness - An OpenClaw-like Inngest-powered personal agent
# Add to your Claude Code skills
git clone https://github.com/inngest/utahLast scanned: 5/30/2026
{
"issues": [],
"status": "PASSED",
"scannedAt": "2026-05-30T16:30:45.787Z",
"npmAuditRan": true,
"pipAuditRan": true
}utah is an open-source ai agents skill for AI coding assistants such as Claude Code, Codex CLI, and ChatGPT, built by inngest. Universally Triggered Agent Harness - An OpenClaw-like Inngest-powered personal agent. It has 118 GitHub stars.
Yes. utah passed SkillsLLM's automated security scan — a dependency vulnerability audit plus prompt-injection heuristics — with no high-severity issues. You can read the full report in the Security Report section on this page.
Clone the repository with "git clone https://github.com/inngest/utah" and add it to your Claude Code skills directory (see the Installation section above).
utah is primarily written in TypeScript. It is open-source under inngest on GitHub, so you can review or fork the full source.
Yes. SkillsLLM lists many other AI Agents skills you can browse and compare side by side. Open the AI Agents category from the badge at the top of this page, or use the Related Skills and comparison links further down to weigh utah against similar tools.
No comments yet. Be the first to share your thoughts!
Universally Triggered Agent Harness
A durable AI agent built with Inngest and pi-ai. No framework. Just a think/act/observe loop — Inngest provides durability, retries, and observability, while pi-ai provides a unified LLM interface across providers.
Simple TypeScript that gives you:
connect(), no server neededChannel (e.g. Telegram) → Inngest Cloud (webhook + transform) → WebSocket → Local Worker → LLM (Anthropic/OpenAI/Google) → Reply Event → Channel API
The worker connects to Inngest Cloud via WebSocket. No public endpoint. No ngrok. No VPS. Messages flow through Inngest as events, and the agent processes them locally with full filesystem access.
git clone https://github.com/inngest/utah
cd utah
npm install # or pnpm
cp .env.example .env
Edit .env with your keys:
ANTHROPIC_API_KEY=sk-ant-...
INNGEST_EVENT_KEY=...
INNGEST_SIGNING_KEY=signkey-prod-...
Then add the environment variables for your channel(s) — see setup guides below.
Start the worker:
# Production mode (connects to Inngest Cloud via WebSocket)
npm start
# Development mode (uses local Inngest dev server)
npx inngest-cli@latest dev &
npm run dev
On startup, the worker automatically sets up webhooks and transforms for each configured channel.
The agent supports multiple messaging channels. Each channel has its own setup guide:
src/
├── worker.ts # Entry point — connect() or serve()
├── client.ts # Inngest client
├── config.ts # Configuration from env vars
├── agent-loop.ts # Core think → act → observe cycle
├── setup.ts # Channel setup orchestration
├── lib/
│ ├── llm.ts # pi-ai wrapper (multi-provider: Anthropic, OpenAI, Google)
│ ├── tools.ts # Tool definitions (TypeBox schemas) + execution
│ ├── context.ts # System prompt builder with workspace file injection
│ ├── session.ts # JSONL session persistence
│ ├── memory.ts # File-based memory system (daily logs + distillation)
│ └── compaction.ts # LLM-powered conversation summarization
├── functions/
│ ├── message.ts # Main agent function (singleton + cancelOn)
│ ├── send-reply.ts # Channel-agnostic reply dispatch
│ ├── acknowledge-message.ts # Message acknowledgment (typing indicator, etc.)
│ ├── heartbeat.ts # Cron-based memory maintenance
│ └── failure-handler.ts # Global error handler with notifications
└── channels/
├── types.ts # ChannelHandler interface
├── index.ts # Channel registry
├── setup-helpers.ts # Inngest REST API helpers for webhook setup
└── <channel-name>/ # A channel implementation (see README for setup)
├── handler.ts # ChannelHandler implementation
├── api.ts # API client
├── setup.ts # Webhook setup automation
├── transform.ts # Webhook transform
└── format.ts # Formatting for channel messages
workspace/ # Agent workspace (persisted across runs)
├── SOUL.md # Agent personality and behavioral guidelines
├── USER.md # User information
├── MEMORY.md # Long-term memory (agent-writable)
├── memory/ # Daily logs (YYYY-MM-DD.md, auto-managed)
└── sessions/ # JSONL conversation files (gitignored)
The core is a while loop where each iteration is an Inngest step:
step.run("think") calls the LLM via pi-ai's complete()step.run("tool-read")Inngest auto-indexes duplicate step IDs in loops (think:0, think:1, etc.), so you don't need to track iteration numbers in step names.
One incoming message triggers multiple independent functions:
| Function | Purpose | Config |
|---|---|---|
agent-handle-message |
Run the agent loop | Singleton per chat, cancel on new message |
acknowledge-message |
Show "typing..." immediately | No retries (best effort) |
send-reply |
Format and send the response | 3 retries, channel dispatch |
agent-heartbeat |
Distill daily logs into long-term memory | Cron (every 30 min) |
global-failure-handler |
Catch errors, notify user | Triggered by inngest/function.failed |
The agent reads markdown files from the workspace directory and injects them into the system prompt:
| File | Purpose |
|---|---|
SOUL.md |
Agent personality, behavioral guidelines, tone, boundaries |
USER.md |
Info about the user (name, timezone, preferences) |
MEMORY.md |
Curated long-term memory (agent-writable) |
Edit these files to customize your agent's personality and knowledge. The agent can also update MEMORY.md using the write tool to remember things across conversations.
The agent has a two-tier memory system:
workspace/memory/YYYY-MM-DD.md) — append-only notes written via the remember tool during conversationsworkspace/MEMORY.md) — curated summary distilled from daily logs by the heartbeat functionThe agent-heartbeat function runs on a cron schedule (default: every 30 minutes). It checks if daily logs have accumulated enough content, then uses the LLM to distill them into MEMORY.md. Old daily logs are pruned after a configurable retention period (default: 30 days).
Long conversations get summarized automatically so the agent doesn't lose context or hit token limits:
Compaction runs as an Inngest step (step.run("compact")), so it's durable and retryable.
Long tool results bloat the conversation context and cause the LLM to lose focus. The agent uses two-tier pruning:
The agent is channel-agnostic. Each channel implements a ChannelHandler interface (src/channels/types.ts) with methods for sending replies, acknowledging messages, and setup. Each channel directory follows the same structure:
src/channels/<name>/
├── handler.ts # ChannelHandler implementation (sendReply, acknowledge)
├── api.ts # API client for the channel's platform
├── setup.ts # Webhook setup automation
├── transform.ts # Plain JS transform for Inngest webhook
└── format.ts # Markdown → channel-specific format conversion
To add Discord, WhatsApp, or any other channel:
src/channels/ following the structure aboveChannelHandler interface in handler.tsagent.message.receivedsrc/channels/index.tsThe agent loop, reply dispatch, and acknowledgment functions are all channel-agnostic — no changes needed outside src/channels/.
connect()](https://www.inngest.com/doc