forgeomni / superagent
AI Agent SDK โ standalone CLI or Laravel integration. Run the full agentic loop in-process with tool use support.
Requires
- php: ^8.1
- guzzlehttp/guzzle: ^7.0
- psr/log: ^3.0
Requires (Dev)
- illuminate/support: ^10.0|^11.0|^12.0
- orchestra/testbench: ^8.0|^9.0|^10.0
- phpunit/phpunit: ^10.0|^11.0
- symfony/console: ^6.0|^7.0
Suggests
- illuminate/support: Required for Laravel service provider, facades, and Artisan commands
- symfony/console: Required for standalone CLI mode (superagent command)
- dev-main
- v1.0.9
- v1.0.8
- v1.0.7
- v1.0.6
- v1.0.5
- v1.0.1
- v1.0.0
- v0.9.9
- v0.9.8
- v0.9.7
- v0.9.6
- v0.9.5
- v0.9.2
- v0.9.1
- v0.9.0
- v0.8.9
- v0.8.8
- v0.8.7
- v0.8.6
- v0.8.5
- v0.8.2
- v0.8.1
- 0.8.0
- v0.7.9
- v0.7.8
- 0.7.7
- v0.7.6
- v0.7.5
- v0.7.2
- v0.7.1
- v0.7.0
- v0.6.19
- v0.6.18
- v0.6.17
- v0.6.16
- v0.6.15
- v0.6.12
- 0.6.11
- 0.6.10
- v0.6.9
- v0.6.8
- v0.6.7
- v0.6.6
- v0.6.5
- v0.6.2
- v0.6.1
- v0.6.0
- v0.5.7
- v0.5.6
- v0.5.5
- v0.5.2
- v0.5.1
- v0.5.0
- dev-feat/opus-4-8-dynamic-workflows
- dev-cli
- dev-copilot/update-tag-to-v082
This package is auto-updated.
Last update: 2026-05-29 03:38:38 UTC
README
๐ Language: English | ไธญๆ | Franรงais ๐ Docs: Installation ยท ๅฎ่ฃ ยท Installation FR ยท Advanced usage ยท API docs
An AI agent SDK for PHP โ run the full agentic loop (LLM turn โ tool call โ tool result โ next turn) in-process, with fourteen providers, real-time streaming, multi-agent orchestration, and a machine-readable wire protocol. Usable as a standalone CLI or as a Laravel library.
superagent "fix the login bug in src/Auth/"
$agent = new SuperAgent\Agent([ 'provider' => 'openai-responses', 'model' => 'gpt-5', ]); $result = $agent->run('Summarise docs/ADVANCED_USAGE.md in one paragraph'); echo $result->text();
Table of Contents
- Quick Start
- Providers & Authentication
- OpenAI Responses API
- Cross-provider handoff
- DeepSeek V4
- Goal mode (codex
/goalparity) - Operational guardrails
- Companion tools (jcode-inspired)
- Agent Loop
- Tools & Multi-Agent
- Agent Definitions
- Skills
- MCP Integration
- Wire Protocol
- Retry, Errors & Observability
- Guardrails & Checkpoints
- Standalone CLI
- Laravel Integration
- Configuration reference
Every feature section ends with a Since line pointing at the release that introduced it. Full release notes live in CHANGELOG.md.
Quick Start
Install:
# As a standalone CLI: composer global require forgeomni/superagent # Or as a Laravel dependency: composer require forgeomni/superagent
See INSTALL.md for the full matrix (system requirements, auth setup, IDE bridges, CI integration).
Smallest possible agent run:
$agent = new SuperAgent\Agent(['provider' => 'anthropic']); $result = $agent->run('what day is it?'); echo $result->text();
Smallest agent run with tools:
$agent = (new SuperAgent\Agent(['provider' => 'openai'])) ->loadTools(['read', 'write', 'bash']); $result = $agent->run('inspect composer.json and tell me what PHP version this project targets'); echo $result->text();
One-shot via CLI:
export ANTHROPIC_API_KEY=sk-... superagent "inspect composer.json and tell me what PHP version this project targets"
Providers & Authentication
Fourteen registry-backed providers, with region-aware base URLs and multiple auth modes per provider. All implement the same LLMProvider contract, so swapping one for another is one line.
| Registry key | Provider | Notes |
|---|---|---|
anthropic |
Anthropic | API key or stored Claude Code OAuth |
openai |
OpenAI Chat Completions (/v1/chat/completions) |
API key, OPENAI_ORGANIZATION / OPENAI_PROJECT |
openai-responses |
OpenAI Responses API (/v1/responses) |
Dedicated section below |
openrouter |
OpenRouter | API key |
gemini |
Google Gemini | API key |
kimi |
Moonshot Kimi | API key; regions intl / cn / code (OAuth) |
qwen |
Alibaba Qwen (OpenAI-compat default) | API key; regions intl / us / cn / hk / code (OAuth + PKCE) |
qwen-native |
Alibaba Qwen (DashScope-native body) | Kept for parameters.thinking_budget callers |
glm |
BigModel GLM | API key; regions intl / cn |
minimax |
MiniMax | API key; regions intl / cn |
deepseek |
DeepSeek V4 | API key; upstreams deepseek / beta / cn / nvidia_nim / fireworks / novita / openrouter / sglang (since v0.9.6, multi-upstream v0.9.8) |
grok |
xAI Grok | API key (XAI_API_KEY / GROK_API_KEY); OpenAI-compatible at api.x.ai; default grok-4.3 (since v1.0.8) |
bedrock |
AWS Bedrock | AWS SigV4 |
ollama |
Local Ollama daemon | No auth โ localhost:11434 by default |
lmstudio |
Local LM Studio server | Placeholder auth โ localhost:1234 by default (since v0.9.1) |
Auth options, by priority:
- API key from environment โ
ANTHROPIC_API_KEY,OPENAI_API_KEY,KIMI_API_KEY,QWEN_API_KEY,GLM_API_KEY,MINIMAX_API_KEY,DEEPSEEK_API_KEY,XAI_API_KEY,OPENROUTER_API_KEY,GEMINI_API_KEY. - Stored OAuth credentials at
~/.superagent/credentials/<name>.json. Device-code flow โ runsuperagent auth login <name>:claude-codeโ reuses an existing Claude Code logincodexโ reuses a Codex CLI logingeminiโ reuses a Gemini CLI loginkimi-codeโ RFC 8628 device flow againstauth.kimi.com(since v0.9.0)qwen-codeโ device flow with PKCE S256 + per-accountresource_url(since v0.9.0)
- Explicit config โ
api_key/access_token/account_idon the agent options.
OAuth refresh is serialised across processes via CredentialStore::withLock() โ parallel queue workers sharing one credential file don't race on refresh (since v0.9.0).
Declarative headers
new Agent([ 'provider' => 'openai', 'env_http_headers' => [ 'OpenAI-Project' => 'OPENAI_PROJECT', // sent only when env set + non-empty 'OpenAI-Organization' => 'OPENAI_ORGANIZATION', ], 'http_headers' => [ 'x-app' => 'my-host-app', // static header ], ]);
Since v0.9.1
Model catalog
Every provider ships with model-id + pricing metadata bundled in resources/models.json. Refresh to the vendor's live /models endpoint at any time:
superagent models refresh # every provider with env creds superagent models refresh openai # one provider superagent models list # show merged catalog superagent models status # catalog source + age
Since v0.9.0
OpenAI Responses API
Dedicated provider at provider: 'openai-responses'. Hits /v1/responses with the full modern OpenAI shape.
Why use it over openai:
| Feature | Responses | Chat Completions |
|---|---|---|
previous_response_id continuation |
โ โ server holds state; new turn skips resending context | โ โ must re-send messages[] every turn |
reasoning.effort (minimal / low / medium / high / xhigh) |
โ native | โ requires model-id hacks for o-series |
reasoning.summary |
โ native | โ |
prompt_cache_key (server-side cache pinning) |
โ native | โ |
text.verbosity (low / medium / high) |
โ native | โ |
service_tier (priority / default / flex / scale) |
โ native | โ |
| Classified error types | โ
via response.failed event codes |
Pattern-matched on HTTP body |
$agent = new Agent([ 'provider' => 'openai-responses', 'model' => 'gpt-5', ]); $result = $agent->run('analyse this codebase and propose refactors', [ 'reasoning' => ['effort' => 'high', 'summary' => 'auto'], 'verbosity' => 'low', 'prompt_cache_key' => 'session:42', 'service_tier' => 'priority', 'store' => true, // required to use previous_response_id next turn ]); // Continue the conversation without resending history: $provider = $agent->getProvider(); $nextAgent = new Agent([ 'provider' => 'openai-responses', 'options' => ['previous_response_id' => $provider->lastResponseId()], ]); $nextResult = $nextAgent->run('now go one level deeper on the auth layer');
ChatGPT subscription routing
Pass access_token (or set auth_mode: 'oauth') to auto-route through chatgpt.com/backend-api/codex โ so Plus / Pro / Business subscribers bill against their subscription instead of getting rejected at api.openai.com.
new Agent([ 'provider' => 'openai-responses', 'access_token' => $token, 'account_id' => $accountId, // adds chatgpt-account-id header ]);
Azure OpenAI
Six base-URL markers auto-flip the provider into Azure mode. api-version query string is added (default 2025-04-01-preview, overridable); api-key header is set alongside Authorization.
new Agent([ 'provider' => 'openai-responses', 'base_url' => 'https://my-resource.openai.azure.com/openai/deployments/gpt-5', 'api_key' => $azureKey, 'azure_api_version' => '2024-12-01-preview', // optional override ]);
Trace-context passthrough
Inject W3C traceparent into client_metadata so OpenAI-side logs correlate with your distributed trace:
$tc = SuperAgent\Support\TraceContext::fresh(); // mint fresh // OR: SuperAgent\Support\TraceContext::parse($headerValue); // from incoming HTTP header $agent->run($prompt, ['trace_context' => $tc]); // OR: $agent->run($prompt, ['traceparent' => '00-0af7-...', 'tracestate' => 'v=1']);
Since v0.9.1
Cross-provider handoff
Agent::switchProvider($name, $config, $policy) swaps the active provider mid-conversation. The message history is preserved and re-encoded into the new provider's wire format on the next request โ so a tool history that ran against Claude can continue under Kimi without losing parallel tool calls or tool_use_id correlation.
use SuperAgent\Conversation\HandoffPolicy; $agent = new Agent(['provider' => 'anthropic', 'api_key' => $key, 'model' => 'claude-opus-4-7']); $agent->run('analyse this codebase'); // Hand off to a cheaper / faster model for the next phase: $agent->switchProvider('kimi', ['api_key' => $kimiKey, 'model' => 'kimi-k2-6']) ->run('write the unit tests'); // Token-window check after switching โ different tokenizers count // the same history differently (Anthropic vs GPT-4 drift 20โ30%): $status = $agent->lastHandoffTokenStatus(); if ($status !== null && ! $status['fits']) { // Trigger your existing IncrementalContext compression before the next call. }
Handoff policy
HandoffPolicy::default() // keep tool history, drop signed thinking, append handoff marker HandoffPolicy::preserveAll() // keep everything โ useful when swap is temporary and you'll come back HandoffPolicy::freshStart() // collapse history to (latest user turn) โ fresh shot at a stuck conversation
Provider-only artifacts the new wire shape can't carry (Anthropic signed thinking, Kimi prompt_cache_key, Responses-API encrypted reasoning, Gemini cachedContent refs) get parked under AssistantMessage::$metadata['provider_artifacts'][$providerKey] โ HandoffPolicy::preserveAll() keeps them around so a later swap back to the originating family can re-stitch them; default() keeps them stashed but invisible to the new provider.
Atomic swap
switchProvider() constructs the new provider before mutating any state. If construction fails (missing api_key, unknown region, network probe rejection) the agent stays on the old provider with its history untouched.
Six wire-format families share one Transcoder
All conversion goes through Conversation\Transcoder, which dispatches by WireFamily enum: Anthropic (also bedrock's anthropic.* invocations), OpenAIChat (OpenAI/Kimi/GLM/MiniMax/Qwen/OpenRouter/LMStudio), OpenAIResponses, Gemini (the only family that correlates tool calls by name+order, no ids), DashScope, Ollama. Useful directly for offline transcoding:
use SuperAgent\Conversation\Transcoder; use SuperAgent\Conversation\WireFamily; $wire = (new Transcoder())->encode($messages, WireFamily::Gemini);
Since v0.9.5
DeepSeek V4
DeepSeek V4 (released 2026-04-24) ships two MoE models โ deepseek-v4-pro (1.6T total / 49B active) and deepseek-v4-flash (284B / 13B active) โ with 1M context as the default and a single-model thinking / non-thinking toggle. The same backend exposes both an OpenAI-wire and an Anthropic-wire endpoint, so the SDK supports two routes:
// OpenAI-wire: native DeepSeekProvider $agent = new Agent([ 'provider' => 'deepseek', 'api_key' => getenv('DEEPSEEK_API_KEY'), 'model' => 'deepseek-v4-pro', // or 'deepseek-v4-flash' ]); // Anthropic-wire: reuse AnthropicProvider with a custom base_url $agent = new Agent([ 'provider' => 'anthropic', 'api_key' => getenv('DEEPSEEK_API_KEY'), 'base_url' => 'https://api.deepseek.com/anthropic', 'model' => 'deepseek-v4-pro', ]);
Reasoning channel. V4-thinking, R1, Kimi-thinking, Qwen-reasoning and any future OpenAI-compat reasoner stream their internal monologue on delta.reasoning_content. The shared ChatCompletionsProvider SSE parser now surfaces it as a separate ContentBlock::thinking() block prepended to the assistant turn โ callers render or hide it deliberately rather than mixing it into the user-facing answer.
$result = $agent->run('hard reasoning prompt', ['thinking' => true]); foreach ($result->message()->content as $block) { if ($block->type === 'thinking') { // model's reasoning chain } elseif ($block->type === 'text') { // user-facing answer } }
Deprecation lane. deepseek-chat and deepseek-reasoner retire 2026-07-24. The catalog flags both with deprecated_until and replaced_by fields; ModelResolver emits a one-shot warning per process recommending deepseek-v4-flash / deepseek-v4-pro respectively. Set SUPERAGENT_SUPPRESS_DEPRECATION=1 to silence.
Cache-aware billing. OpenAI-compat backends report prompt_tokens as gross (cache hits + misses). The parser now subtracts the cached portion before populating Usage::inputTokens, so the cache discount lands correctly โ CostCalculator charges 10% of input price for read hits instead of effectively 110%. Affects every OpenAI-compat backend with caching (DeepSeek, Kimi, OpenAI itself).
Beta endpoint. Set region: 'beta' to route to https://api.deepseek.com/beta for FIM / prefix completion access on the same auth โ see completeFim() for the dedicated helper.
Since v0.9.6
Reasoning-effort dial (v0.9.8)
Three-tier dial across DeepSeek native + every relay:
// Cheapest: thinking off entirely. $agent->run('translate this paragraph', options: ['reasoning_effort' => 'off']); // Standard thinking budget (V4-Pro tier default). $agent->run('design a queue with at-least-once semantics', options: ['reasoning_effort' => 'high']); // Deepest CoT โ V4-Pro "think harder". Slower, more expensive. $agent->run('audit this migration for race conditions', options: ['reasoning_effort' => 'max']);
Each upstream gets the body shape it expects: top-level
reasoning_effort + thinking: {type: enabled} for DeepSeek native /
OpenRouter / Novita / Fireworks / SGLang; nested
chat_template_kwargs.{thinking, reasoning_effort} for NVIDIA NIM.
Unknown values silently no-op rather than poisoning the request.
Multi-upstream routing (v0.9.8)
Same V4 weights, six relay paths. One upstream config key picks the
host:
$agent = new Agent([ 'provider' => 'deepseek', 'upstream' => 'fireworks', // or nvidia_nim / novita / openrouter / sglang 'options' => ['model' => 'deepseek-v4-pro'], ]); // Self-hosted SGLang requires explicit base_url: $agent = new Agent([ 'provider' => 'deepseek', 'upstream' => 'sglang', 'base_url' => 'http://my-sglang:30000/v1', ]);
region is preserved as an alias of upstream for backward
compatibility โ existing region: 'default' | 'cn' | 'beta' callers
are byte-compatible.
V4 Interleaved-Thinking replay (v0.9.8)
V4 thinking mode rejects assistant messages that carry tool_calls
without reasoning_content. The provider now:
- Re-emits each
AssistantMessage'sthinkingblocks as wirereasoning_contentautomatically (no caller change). - Runs a final-pass sanitizer that forces a
(reasoning omitted)placeholder on any assistant+tool_calls that slipped through โ bullet-proofs sessions restored from disk pre-0.9.8 and sub-agents that hand-build messages.
Disable with reasoning_effort: 'off' (sanitizer skips when thinking
is explicitly disabled).
FIM (prefix completion) (v0.9.8)
$agent = new Agent([ 'provider' => 'deepseek', 'region' => 'beta', ]); $completed = $agent->provider()->completeFim( prefix: "function fibonacci(\$n) {\n ", suffix: "\n}\n", options: ['max_tokens' => 64], );
Hits https://api.deepseek.com/beta/v1/completions. Throws when the
provider isn't on the beta region rather than silently routing
elsewhere.
/model auto heuristic (v0.9.8)
use SuperAgent\Routing\AutoModelStrategy; $strategy = new AutoModelStrategy(); $model = $strategy->select($messages, $systemPrompt, $options); // โ 'deepseek-v4-pro' or 'deepseek-v4-flash' $agent = new Agent([ 'provider' => 'deepseek', 'options' => ['model' => $model, 'reasoning_effort' => 'high'], ]);
Pro escalation when: prompt โฅ 32K tokens, โฅ 3 trailing tool turns,
explicit reasoning_effort=max, or system-prompt keywords
(review / audit / design / architect / plan / debug a complex / analyze the codebase / find the root cause). Flash otherwise.
Cache-aware compaction (v0.9.8)
use SuperAgent\Context\Strategies\CacheAwareCompressor; use SuperAgent\Context\Strategies\ConversationCompressor; $compactor = new CacheAwareCompressor( delegate: new ConversationCompressor($estimator, $config, $provider), tokenEstimator: $estimator, config: $config, pinHead: 4, // first 4 messages stay byte-stable pinSystem: true, // also pin the system message );
Wraps any CompressionStrategy. Result shape:
[head_pinned, summary_boundary, summary, tail_preserved] with the
cached prefix at byte 0. Idempotent across rounds โ feeding a
compacted result back through the wrapper preserves the same prefix
bytes, so DeepSeek's auto prefix cache keeps hitting on every
/compact.
Goal mode (codex /goal parity) (v0.9.8)
Three model-callable tools, four-state lifecycle, two prompt
templates. Goals are thread-scoped; each thread has at most one
non-terminal goal at a time. The model can ONLY transition active โ complete; pause / resume / budget changes flow from user / system.
use SuperAgent\Goals\GoalManager; use SuperAgent\Goals\InMemoryGoalStore; use SuperAgent\Tools\Builtin\CreateGoalTool; use SuperAgent\Tools\Builtin\GetGoalTool; use SuperAgent\Tools\Builtin\UpdateGoalTool; $threadId = 'session-42'; $goals = new GoalManager(new InMemoryGoalStore()); $agent->registerTool(new CreateGoalTool($goals, $threadId)); $agent->registerTool(new GetGoalTool($goals, $threadId)); $agent->registerTool(new UpdateGoalTool($goals, $threadId)); // On each turn, account tokens and inject continuation when idle: $agent->onTurnEnd(function ($usage) use ($goals, $threadId) { $goal = $goals->getActive($threadId); if ($goal === null) return; $updated = $goals->recordUsage($goal->id, $usage->inputTokens + $usage->outputTokens); if ($updated->status === GoalStatus::BudgetLimited) { $agent->injectSystemMessage($goals->renderBudgetLimitPrompt($updated)); } elseif ($updated->status === GoalStatus::Active) { $agent->injectSystemMessage($goals->renderContinuationPrompt($updated)); } });
Persistence. InMemoryGoalStore ships with the SDK; SuperAICore
provides EloquentGoalStore (table ai_goals) so a goal survives
process restarts.
Untrusted-input wrapping. Both prompt templates wrap the user
objective in <untrusted_objective> via Security\UntrustedInput::tag()
so a crafted goal can't smuggle higher-priority instructions into the
system role:
use SuperAgent\Security\UntrustedInput; $wrapped = UntrustedInput::wrap($userInput, kind: 'note'); // โ "The text below is user-provided data..." + "<untrusted_note>...</untrusted_note>"
Recommended at every site that injects user-supplied text into a system-role message โ goals, skills, memory imports.
Operational guardrails (v0.9.8)
Sub-agent depth cap
Cap on recursive agent tool calls. Mirrors codex's agents.max_depth.
use SuperAgent\Swarm\AgentDepthGuard; // Set the cap (default 5; env: SUPERAGENT_MAX_AGENT_DEPTH). AgentDepthGuard::setMax(8); // In the spawn site, before launching the child: AgentDepthGuard::check(); // throws AgentDepthExceededException at cap $childEnv = AgentDepthGuard::forChild(); // pass to proc_open / Symfony\Process
Depth tracked through the SUPERAGENT_AGENT_DEPTH env so it survives
process spawning.
Token-bucket rate limiter
DeepSeek-TUI shape (8 RPS sustained, 16-burst):
use SuperAgent\Providers\Transport\TokenBucket; $bucket = new TokenBucket(ratePerSecond: 8.0, burst: 16); $bucket->consume(); // blocks until capacity if (! $bucket->tryConsume()) { /* skip / queue */ }
In-process fidelity. Cross-process limits are a host concern (Redis- backed Guzzle middleware).
Ephemeral conversation fork (/side semantics)
use SuperAgent\Conversation\Fork; $fork = Fork::from($parentMessages); $fork->extend(new UserMessage('try the alternative approach'), $sideAssistantReply); // Either discard or promote selected side messages back into parent: $parentNext = $fork->discard(); // throw the side away $parentNext = $fork->promote(2); // bring back side message #2 only $parentNext = $fork->promoteAll(); // bring everything back
Ad-hoc memory injection
use SuperAgent\Memory\AdHocMemoryProvider; $adhoc = new AdHocMemoryProvider(); $adhoc->push('CI is currently red on main', ttlSeconds: 1800, untrusted: true); $adhoc->push('You MUST output JSON', ttlSeconds: 0, untrusted: false); // sticky + trusted $memoryManager->setExternalProvider($adhoc); // Next turn sees both entries via onTurnStart(); ad-hoc is push-only โ // search() returns []. Compose alongside BuiltinMemoryProvider, not in place of.
Companion tools (jcode-inspired)
Five additive primitives borrowed from jcode. Each is opt-in and degrades to no-op when its host wiring is absent.
agent_grep โ token-aware grep with enclosing-symbol injection
A sibling of the byte-for-byte ripgrep grep tool. Same flags, plus per-match enclosing-symbol metadata (PHP / JS / TS / Python / Go) and per-session seen-chunk truncation so the model doesn't re-read the same hunk three turns in a row.
$agent->loadTools(['grep', 'agent_grep']); // both registered, pick per call // Default: regex-based extractor (dependency-free, ~95% accuracy) $agent->run('find every caller of MyClass::handle and show me which method contains it');
Symbol extraction is pluggable via the Tools\Builtin\Symbols\SymbolExtractor SPI:
use SuperAgent\Tools\Builtin\AgentGrepTool; use SuperAgent\Tools\Builtin\Symbols\CompositeSymbolExtractor; use SuperAgent\Tools\Builtin\Symbols\TreeSitterSymbolExtractor; use SuperAgent\Tools\Builtin\Symbols\RegexSymbolExtractor; $agent->registerTool(new AgentGrepTool(symbolExtractor: new CompositeSymbolExtractor([ new TreeSitterSymbolExtractor(), // shells out to `tree-sitter` CLI; ~15 grammars new RegexSymbolExtractor(), // pure-PHP fallback; always works ])));
Tree-sitter is auto-discovered on $PATH (override via SUPERAGENT_TREE_SITTER_BIN or constructor arg). Missing binary / unsupported grammar / failed invocation degrades to "I don't support this" โ never throws.
FileLedger โ cross-agent edit notification for swarms
Agent A edits a file that agent B has read; B gets a FileShiftedEvent in its mailbox. Lazy-attached to WorktreeManager::fileLedger(), opt-in by tools that record reads/writes; default emitter is no-op so existing swarms are byte-compatible.
$ledger = $worktreeManager->fileLedger(); $ledger->setEmitter(function (FileShiftedEvent $event, string $toAgent) { // event = {path, byAgent, at, summary, shaBefore, shaAfter} $mailbox->push($toAgent, $event); }); $ledger->recordRead($agentB, '/abs/file.php'); $ledger->recordWrite($agentA, '/abs/file.php', shaBefore: '...', shaAfter: '...', summary: 'fixed null guard'); // โ emitter fires with toAgent=$agentB
AmbientWorker โ background memory hygiene with cost split
Long-lived low-priority worker that runs memory dedup + staleness scans on a tick. Tick budget enforced internally so a pass never blocks for more than a few seconds. Token cost is tagged usage_source: 'ambient' via the supplied callback so dashboards split user-facing vs background spend.
$worker = new AmbientWorker( memoryProvider: $memProvider, usageReporter: fn(Usage $u) => $costMeter->record($u, source: 'ambient'), passBudgetSeconds: 3, ); while ($host->running()) { $worker->tick(); // call from cron, swoole, react, or plain `while sleep` sleep(60); }
Native browser bridge (Firefox / Chromium)
WebExtension Native Messaging โ 4-byte length-prefixed JSON framing โ lets an agent drive a real browser without Selenium / Playwright. Single launcher per tool instance; tight capability surface (no tab management, cookies, or extension APIs).
$agent->registerTool(new FirefoxBridgeTool()); $agent->run('open https://example.com, take a screenshot, click the "Sign in" link, screenshot again');
Launcher path comes from SUPERAGENT_BROWSER_BRIDGE_PATH (or constructor launcherArgv). The companion Tools\Browser\FirefoxBridge::class docblock contains the full WebExtension + Native Messaging manifest walkthrough.
Pluggable embeddings โ Memory\Embeddings\*
EmbeddingProvider interface (batch shape, dimensions(), fingerprint()). Three reference implementations:
| Class | Path of least resistance for |
|---|---|
OllamaEmbeddingProvider |
Devs already running Ollama locally โ talks to /api/embeddings, default nomic-embed-text (768 dims) |
OnnxEmbeddingProvider |
In-process inference โ needs ext-onnxruntime or ankane/onnxruntime + a model file |
NullEmbeddingProvider |
Tests / dev โ returns []; downstream falls back to keyword scoring |
CallableEmbeddingProvider |
Adapts existing fn(array): array or legacy fn(string): array<float> closures |
Hooks straight into the upgraded SemanticSkillRouter:
use SuperAgent\Skills\SemanticSkillRouter; use SuperAgent\Memory\Embeddings\OllamaEmbeddingProvider; $router = new SemanticSkillRouter( embedder: new OllamaEmbeddingProvider(), // or any EmbeddingProvider topK: 5, ); // Falls back to keyword overlap when no embedder; vector cache keyed by skill content hash.
superagent resume โ cross-harness session pickup
Pick up a Claude Code or Codex CLI session in SuperAgent without losing the thread.
superagent resume list --from claude
superagent resume show --from claude --session 8e2c-...
superagent resume load --from claude --session 8e2c-... \
| superagent chat --provider kimi --resume-stdin
--from accepts claude / claude-code / cc / codex. Behind the scenes: Conversation\HarnessImporter interface + per-harness importers (ClaudeCodeImporter reads ~/.claude/projects/<hash>/<uuid>.jsonl; CodexImporter reads ~/.codex/sessions/**/*.jsonl), feeding internal Message[] into the existing Conversation\Transcoder so the transcript flips wire family transparently.
Since v0.9.7
Agent Loop
Agent::run($prompt, $options) drives the full turn loop until the model stops emitting tool_use blocks. Each turn's cost, usage, and messages flow into AgentResult.
$result = $agent->run('...', [ 'model' => 'claude-sonnet-4-5-20250929', // per-call override 'max_tokens' => 8192, 'temperature' => 0.3, 'response_format' => ['type' => 'json_schema', 'json_schema' => [...]], 'idempotency_key' => 'job-42:turn-7', // since v0.9.1 'system_prompt' => 'You are a precise analyst.', ]); echo $result->text(); $result->turns(); // turn count $result->totalUsage(); // Usage{inputTokens, outputTokens, cache*} $result->totalCostUsd; // float, across all turns $result->idempotencyKey; // passthrough for usage-log dedup (since v0.9.1)
Budget + turn caps
$agent = (new Agent(['provider' => 'openai'])) ->withMaxTurns(50) ->withMaxBudget(5.00); // USD โ hard cap; aborts mid-loop if breached
Streaming
foreach ($agent->stream('...') as $assistantMessage) { echo $assistantMessage->text(); }
For machine-readable event streams (JSON / NDJSON for IDE / CI consumers) see the Wire Protocol section.
Auto-mode (task detection)
new Agent([ 'provider' => 'anthropic', 'auto_mode' => true, // delegates to TaskAnalyzer to pick model + tools ]);
Squad mode โ Adaptive Cross-Model Squad (v0.9.9)
Auto-mode picks single vs. multi-agent. Squad mode goes further: when a prompt decomposes into 2+ subtasks spanning multiple difficulty bands, each subtask is dispatched to its own model โ Haiku for trivial extraction, Sonnet for moderate refactors, DeepSeek-Pro / Opus for hard reasoning. There is no master agent โ the workflow definition is the orchestrator and every step is a peer. Human-in-the-Loop gates sit inline as ApprovalSteps.
use SuperAgent\Squad\{TaskDecomposer, PeerOrchestrator, SquadCheckpointStore}; $subTasks = (new TaskDecomposer())->decompose( "1. Research the auth module\n". "2. Architect a migration plan (ๆฒๅฎๆนๆก้่ฆไบบๅทฅๅฎกๆ ธ)\n". "3. Implement OAuth2" ); $orchestrator = new PeerOrchestrator( agentDispatcher: $myDispatcher, // (SquadDispatchRequest $r) => string|array checkpointStore: new SquadCheckpointStore('/tmp/cp'), // per-step crash recovery output: $consoleOutput, // streams progress events maxCostUsd: 5.00, // downshift remaining steps at 80% of cap ); $result = $orchestrator->run('refactor-2026-05', $subTasks);
What it gives you that the master-slave path doesn't:
- Cross-model: each subtask picks its own (provider, model) via
ModelTierMap. Defaults are intentionally cross-vendor (Anthropic + DeepSeek + โฆ) so a HARD subtask doesn't pay Opus rates for an EASY peer. - Stable per-role sessions:
squad:{squadId}:role:{roleName}is reused across resumes โ the provider's prompt-cache prefix survives, so re-running step N doesn't re-prime the model. - Skip + restart:
SquadResumeManagerre-seeds completed step outputs and BFS-invalidates only what depends on a restarted step. - Parallel groups: prompts with
ๅๆถ / in parallelare split onๅ / and, and peers in the same group execute through a singleParallelStep. - Cost downshift: at 80% of
maxCostUsd, remaining steps drop one tier (EXPERT โ HARD โ MODERATE โ โฆ). - Provider fallback ladder: if a band's primary provider isn't registered,
ModelTierMap::resolve()walks down then up to a registered alternative. - Auto-trigger from auto mode:
AutoModeAgentroutes into squad when the decomposed prompt spans โฅ 2 difficulty bands. Force withsuperagent auto "<task>" --squad, opt out with--no-squad. - Peer-to-peer messaging: agents talk directly via
PeerMailbox(tell/broadcast/ask) โ no master agent relaying summaries.PeerAskroutes through the peer's stable session so its prompt cache survives. Read-onlyPeerAsk/PeerSend/PeerInboxtools let an agent call peers from inside its own tool loop.
CLI:
superagent auto "<task>" --squad --max-cost 5.00 --verbose
Config (config/superagent.php):
'squad' => [ 'prefer_squad' => true, 'max_cost_usd' => 5.00, 'checkpoint_dir' => '/var/lib/superagent/squad', 'tier_map' => [ // override any/all bands; defaults fill the rest 'expert' => ['provider' => 'openai', 'model' => 'gpt-5-pro'], ], ],
YAML team library (v1.0.1)
A SquadPlan no longer requires PHP code. Drop a YAML file, register it, run it:
# resources/squad-teams/code-review-loop.yaml (bundled) name: code-review-loop description: Writer + reviewer with feedback-injection loop tier_map: hard: {provider: anthropic, model: claude-opus-4-7} expert: {provider: openai, model: gpt-5.1-codex} steps: - name: write difficulty: hard prompt: "{{task}}" - name: review difficulty: expert depends_on: [write] prompt: "Artefact:\n{{steps.write.output}}" pause_after: true # optional HITL gate loops: - writer: write reviewer: review feedback_key: review.feedback max_retries: 3
use SuperAgent\Squad\TeamRegistry; $plan = (new TeamRegistry())->require('code-review-loop'); $result = $orchestrator->run('refactor-2026-05', $plan->subTasks, /* โฆ */);
21 production-grade teams ship in resources/squad-teams/ โ engineering (code-review-loop, code-bug-triage, code-test-driven, code-security-audit, code-perf-optimize, โฆ), architecture (arch-from-scratch, arch-decision-record, arch-migration-plan), QA/SRE (qa-ship-gate, qa-multi-model-council, qa-incident-response), product (product-strategy-trio, product-discovery-pair), docs (docs-tech-pipeline, docs-api-spec-pipeline), data (data-research-trio, data-anomaly-detector), ops/growth (release-coordination, growth-hypothesis-test).
Three-tier override: hosts call addDirectory('/path/to/host-teams') to layer additional YAML files on top of the bundled library, or register($name, $plan) for programmatic overrides. Later directories override earlier ones; runtime registrations override directory entries. Same pattern ModelCatalog uses.
Reviewer-loop runner: Squad\ReviewerLoopRunner wraps any agentDispatcher callable. When the reviewer's first non-blank line doesn't start with APPROVED (case-insensitive), the runner prepends the reviewer's feedback to the writer's prompt and re-dispatches the writer. Loop exits on approval or max_retries.
Cross-mode orchestration (v1.0.1)
auto / smart / squad can now compose, recurse, and hand off through one shared ModeContext. A squad step can declare mode: smart and recurse into the orchestrator; a smart sub-task can land on a full squad; a reviewer that hits max_retries escalates to smart automatically. Across every layer, blackboard reads, cost-ledger writes, and prompt-cache session ids share one object โ there are no isolated child islands.
steps: - name: research mode: smart # recurse: smart decomposes + routes internally prompt: "Investigate {{task}}" - name: implement mode: squad team: code-review-loop # use a bundled team for this step prompt: "Build based on {{steps.research.output}}" - name: stress-test mode_chain: [single, smart, squad] # escalation chain on failure fail_criteria: "REJECTED|error" prompt: "Stress-test the build"
The ModeContext flows through every recursion level:
use SuperAgent\Modes\{ModeContext, ModeRouter, CrossModePolicy}; $ctx = ModeContext::root('squad', policy: new CrossModePolicy( maxDepth: 4, budgetCapUsd: 5.00, autoEscalateOnFailure: true, escalateTo: 'smart', )); $router = new ModeRouter(); $router->register(new AutoModeAdapter($autoModeAgent)); $router->register(new SmartModeAdapter($smartOrchestrator)); $router->register(new SquadModeAdapter()); $result = $router->dispatch('squad', $task, $ctx); // $ctx->costLedger->total() โ every leaf dispatch summed // $ctx->costLedger->byMode() โ {squad: 0.31, smart: 0.18, auto: 0.02} // $ctx->blackboard->entries() โ every claim / evidence / risk / decision
Loose-coupling SPIs:
Squad\SquadDispatcherRegistry::set($dispatcher)โ hosts install a default squad dispatcher (e.g. a CLI-aware one)Modes\ModeRouterRegistry::set($router)โ hosts install a default cross-mode router (e.g. one that knowscli:claude_clileaf tags alongside the three mode names)
Both are class-level static slots. SDK code paths consult them before falling back to internal defaults. SDK itself never sets either โ slots are reserved for hosts.
Gemini 3.5 + thinking / grounding (v1.0.5)
gemini-3.5-pro / gemini-3.5-flash / gemini-3.5-flash-lite are first-class catalog entries (alias gemini โ 3.5 Pro). The 3.x preview SKUs from gemini-cli (gemini-3-pro-preview, gemini-3.1-pro-preview, gemini-3-flash-preview, gemini-3.1-flash-lite-preview) ship too. Provider default upgraded from gemini-2.0-flash โ gemini-3.5-flash.
$agent = new Agent([ 'provider' => 'gemini', 'model' => 'gemini-3.5-pro', 'options' => [ 'thinking' => \SuperAgent\Thinking\ThinkingConfig::adaptive(), // โ thinkingLevel: HIGH 'grounding' => true, // โ tools[].googleSearch 'url_context' => true, // โ tools[].urlContext ], ]);
AssistantMessage::$metadata['grounding_sources'] carries [{uri, title}, ...] for any Google-Search citation; parts[].thought (Gemini 3.x thinking parts) is surfaced as ContentBlock::thinking(), and usageMetadata.thoughtsTokenCount folds into Usage::$outputTokens.
Semantic loop detection (v1.0.5)
Guardrails\LoopDetector already catches hash-identical repetition (5 tool-call dupes, 10-char content chants, 8 file-reads in a 15-call window, โฆ). New Guardrails\LlmLoopChecker complements it with a Flash-model probe after the 30th turn โ catches semantic loops the hashes miss (model paraphrases the same plan ten times without acting). Prompt is the verbatim gemini-cli LOOP_DETECTION_SYSTEM_PROMPT; check interval adjusts dynamically (5โ15 turns).
$flash = new GeminiProvider(['model' => 'gemini-3.5-flash', 'api_key' => env('GEMINI_API_KEY')]); $checker = new LlmLoopChecker($flash); foreach ($turns as $i => $turn) { if ($v = $checker->turnStarted($i, $conversationHistory, $originalPrompt)) { // $v->type === LoopType::LlmDetected; halt and surface to user } }
Six opencode patterns ported (v1.0.5)
Each fills a gap relative to a production coding agent; every piece is opt-in and tested.
| Module | What it does |
|---|---|
Permissions\BashArity |
110+ CLI arity table (gitโ2, docker composeโ3, terraform workspaceโ3, vault kvโ3, โฆ). BashCommandClassifier::extractPrefix() uses longest-prefix-wins so permission rules match the right granularity. |
Context\Strategies\ConversationCompressor::getStructuredSummaryPrompt() |
7-section Markdown template (Goal / Constraints / ProgressยทDoneยทInProgressยทBlocked / Decisions / Next Steps / Critical Context / Relevant Files). Select via options: ['summary_prompt' => 'structured']. |
Format\ namespace |
26 auto-formatters (gofmt, prettier, biome, ruff, rustfmt, pint, rubocop, shfmt, clang-format, terraform fmt, โฆ). (new FormatterRunner())->formatFile($path, $worktree) runs every applicable formatter after an edit. |
LSP\ namespace |
Real stdio JSON-RPC LSP client (was a 'simulated' stub). 9 servers: phpactor/intelephense, gopls, rust-analyzer, pyright, typescript-language-server, clangd, bash-language-server, zls. LSPTool exposes diagnostics/hover/definition/touch. |
ACP\ namespace |
Agent Client Protocol v1 server. Editors that speak ACP (Zed, Neovim with Codecompanion, โฆ) plug into SuperAgent directly. (new Server($handler))->serve() blocks on stdio. |
Skills\SkillManager::discoverExternalSkills() |
Walks upward from cwd to worktree root loading .claude/skills/**/SKILL.md and .agents/skills/**/SKILL.md at every level; at root, also skills/**/SKILL.md / skill/**/SKILL.md. Files must literally be named SKILL.md. |
Chrome Trace Event timeline (v1.0.6)
Every long-running orchestration (debate, red-team, error-recovery, cost-autopilot, parallel agents) now writes a lock-free ring buffer of TraceEvent records. Trigger sites (errors, snapshot tool calls, completed protocols) flush to a .json file you open in chrome://tracing or ui.perfetto.dev.
use SuperAgent\Tracing\TraceCollector; $end = TraceCollector::getInstance()->span('llm.dispatch', 'llm', 'session:abc'); $result = $provider->call(...); $end(['model' => $result->model, 'cost_usd' => $result->cost]); // At a trigger point โ error, end-of-debate, agent self-snapshot, etc. $path = TraceCollector::getInstance()->dump(trigger: 'manual', reason: 'inspect after demo'); // โ /tmp/superagent-traces/trace_superagent_{session}_{ts}_manual.json
Wired-up emitters: Debate\DebateOrchestrator (debate.start / debate.rounds / debate.round_N / debate.judge / debate.total plus redteam.* / ensemble.*), ErrorRecovery\ErrorRecoveryManager (auto-dumps the last N events on unrecoverable / retries-exhausted), CostAutopilot\CostAutopilot (budget.spend counter + budget.tier_change instants), Console\Output\ParallelAgentDisplay::exportPerfettoJson() (whole-team snapshot). Env: SUPERAGENT_TRACE_ENABLED=false to disable, SUPERAGENT_TRACE_PATH=โฆ to override.
The agent itself can call snapshot('about to git reset', tag: 'pre_destructive') (new SnapshotTool) when something feels off โ pure observability, isReadOnly === true.
Pi-aligned JSON Event Stream (v1.0.6)
A canonical 18-type session-event taxonomy borrowed from pi's JSON Event Stream Mode so SuperAgent sessions can be replayed by any pi-compatible viewer.
use SuperAgent\Tracing\PiEventStream; use SuperAgent\Tracing\PiEventStreamWriter; PiEventStream::subscribe(new PiEventStreamWriter('/path/to/session.events.jsonl')); PiEventStream::emit(PiEventStream::AGENT_START, ['sessionId' => 's-1']); PiEventStream::emit(PiEventStream::TURN_START, ['turnId' => 't-1', 'sessionId' => 's-1', 'model' => 'claude-opus-4-7']); // ... PiEventStream::emit(PiEventStream::AGENT_END, ['sessionId' => 's-1']);
Event types: session / agent_start / agent_end / turn_start / turn_end / message_start / message_update / message_end / tool_execution_start / tool_execution_update / tool_execution_end / queue_update / compaction_start / compaction_end / auto_retry_start / auto_retry_end / model_change / thinking_level_change. Legacy SuperAgent event names round-trip via PiEventStream::translateLegacy().
Mid-turn steering + follow-up queue (v1.0.6)
Pi-borrowed mid-turn correction without aborting + post-turn follow-up queueing. An operator (or host RPC handler) can nudge a running agent without losing tool state.
// While agent is mid-turn, from a separate event handler: $agent->steer('Stop, the bug is in src/Auth/Session.php, not src/Auth/Login.php'); // The next QueryEngine iteration drains the queue and prepends the steer as a synthetic user message. // Or queue a follow-up that fires AFTER the current turn ends: $agent->followUp('Once you finish, also run the tests in tests/Unit/Auth/');
Same surface on ACP: editors that speak the protocol send session/steer / session/follow_up JSON-RPC methods (constants Protocol::METHOD_SESSION_STEER / METHOD_SESSION_FOLLOW_UP). Handler contract gains steer($params) / followUp($params); DefaultHandler ships drain helpers (drainSteer($sessionId) / drainFollowUp($sessionId)) for the host's promptFn to consume at safe checkpoints.
RTK structured-output compression (v1.0.6)
git diff / grep -rn / find / ls -R / tree outputs dominate token usage in coding sessions โ typically 30-50 % of input budget per turn, much of it cosmetic. Tools\Compression\RtkPipeline (borrowed from 9Router + claude-octopus) auto-compresses these in a lossy-safe way: paths, line numbers, diff hunks are preserved verbatim; cosmetic noise is dropped. Wired into QueryEngine โ every non-error tool result goes through it.
| Tool | Typical savings | What it preserves |
|---|---|---|
git diff |
40-65 % | diff --git, --- a/, +++ b/, @@ hunks, every +/- line, mode markers |
grep / rg |
30-50 % | file:line:match canonical hits; compacts repeated paths to indent |
find |
25-40 % | Leaf filenames; collapses repeated directory prefixes |
ls -R, tree |
20-35 % | Structure markers (โโ / โโ / โ); drops byte/size annotations |
Opt out per-call via options: ['disable_rtk_compression' => true]. Stats available via $pipeline->stats() (bytes_in / bytes_out / saved_bytes / ratio).
Cross-provider tool-schema portability (v1.0.6)
Anthropic, OpenAI, Gemini all accept slightly different subsets of JSON Schema (Gemini rejects $ref / $defs / top-level oneOf; OpenAI strict mode is stricter than non-strict; etc.). Tools\Schema\Schema lets you declare a schema once carrying intent markers; ProviderNormalizer rewrites it per provider.
use SuperAgent\Tools\Schema\Schema; use SuperAgent\Tools\Schema\ProviderNormalizer; $inputSchema = Schema::object([ 'mode' => Schema::stringEnum(['read', 'write', 'delete']), 'path' => Schema::string('Absolute path'), ], required: ['mode', 'path']); // In your tool's inputSchema(), rewrite per provider before sending: $anthropic = ProviderNormalizer::forAnthropic($inputSchema); $openai = ProviderNormalizer::forOpenAI($inputSchema); $gemini = ProviderNormalizer::forGemini($inputSchema); // โ Gemini variant has `oneOf` flattened to `enum`, `$ref`/`$defs` removed, unsupported `format` stripped.
Cross-provider compliance is verified by tests/Tools/Schema/CrossProviderComplianceTest.php โ round-trips every tagged shape through all three normalisers and asserts each output passes the target provider's acceptance rules.
Session branching โ pi /tree fork (v1.0.6)
Pi models a session as an append-only tree, not a line. SuperAgent now mirrors /tree:
use SuperAgent\Session\SessionManager; $newBranchId = $sessionManager->fork( sourceSessionId: 's-current', forkAtIndex: 12, summaryFn: function(array $abandoned) use ($llm) { return $llm->summarize($abandoned); // one-line summary stored on the source }, displayName: 'try-event-sourcing-instead', );
Conversation\BranchManager provides the pure tree algebra (no I/O, no LLM) โ leaves(), ancestry($id), findCommonAncestor($a, $b), collectBranch($leaf, $ancestor), makeBranchSummaryEntry(). Conversation\Importers\PiImporter replays existing pi sessions (~/.pi/agent/sessions/) into SuperAgent's wire format.
Squad consensus gates โ N-of-M parallel voting (v1.0.6)
The existing reviewer_loop is a serial, single-veto gate (one reviewer agrees โ pass). Squad\ConsensusGate (borrowed from claude-octopus) is a parallel N-of-M gate: M peers vote in parallel, โฅN must approve.
- kind: consensus_gate name: ship-decision depends_on: [synthesize] params: n: 3 m: 4 voters: - role: qa-bach - role: security-schneier - role: cto-vogels - role: ceo-bezos prompt: | Review the proposed change and respond with one line: VERDICT: APPROVE | REJECT | ABSTAIN followed by a single-paragraph justification.
Tally returns {verdict, passed, counts, per_voter}; downstream steps branch on passed.
Qwen 3.7 Max + Anthropic-protocol drop-in (v1.0.6)
Qwen 3.7 Max (released 2026-05-21) ships native Anthropic API protocol support โ the /v1/messages wire is byte-compatible with Anthropic's canonical shape, so Claude-Code-shaped clients can point at DashScope and use Qwen as a drop-in.
// Drop-in Anthropic-shaped requests against Qwen: $agent = new Agent([ 'provider' => 'qwen-anthropic', 'api_key' => env('DASHSCOPE_API_KEY'), 'model' => 'qwen3.7-max', // 1M ctx, $2.50 / $7.50 per 1M ]); // Or the OpenAI-compat path with the new default model: $agent = new Agent(['provider' => 'qwen']); // โ defaults to qwen3.7-max
TODO: DashScope's exact Anthropic-protocol endpoint URL is not yet documented in English; the default
dashscope.aliyuncs.com/anthropic-mode/v1is best-guess. Verify and override viabase_urluntil Alibaba publishes it.
Recipe cookbook (v1.0.6)
Focused, copy-pasteable recipes for the higher-level subsystems live under docs/cookbook/:
01-debate-protocol.mdโ proposer / critic / judge structured debate02-redteam-attack.mdโ builder / attacker / reviewer adversarial pattern03-cost-autopilot.mdโ budget-driven model tiering (Opus โ Sonnet โ Haiku cascade)04-cost-prediction.mdโ KNN-based proactive spend prediction05-adaptive-feedback.mdโ corrections promote into auto-applied patterns
Idempotency
$result = $agent->run($prompt, ['idempotency_key' => $queueJobId . ':' . $turnNumber]); // $result->idempotencyKey is truncated to 80 chars; surfaces on the AgentResult // so hosts that write ai_usage_logs can dedupe on it.
Since v0.9.1
Tools & Multi-Agent
Tools are subclasses of SuperAgent\Tools\Tool. Built-in tools โ read / write / edit / bash / glob / grep / search / fetch โ auto-load unless the caller opts out. Custom tools register via $agent->registerTool(new MyTool()).
$agent = (new Agent(['provider' => 'anthropic'])) ->loadTools(['read', 'write', 'bash']) ->registerTool(new MyDomainTool()); $result = $agent->run('apply the refactor plan in ./plan.md');
Multi-agent orchestration (AgentTool)
Dispatch sub-agents in parallel by emitting multiple agent tool_use blocks in one assistant message:
$agent->registerTool(new AgentTool()); $result = $agent->run(<<<PROMPT Run these three investigations in parallel: 1. Read CHANGELOG.md and summarise the last three releases 2. Read composer.json and list all runtime dependencies 3. Grep for TODO comments in src/ Collate the three reports. PROMPT);
Each sub-agent runs in its own PHP process (via ProcessBackend); blocking I/O in one child doesn't block siblings. When proc_open is disabled, fibers take over.
Productivity evidence
Every AgentTool result carries hard evidence of what the child actually did โ not just success: true:
[
'status' => 'completed', // or 'completed_empty' / 'async_launched'
'filesWritten' => ['/abs/path/a.md'], // deduped absolute paths
'toolCallsByName' => ['Read' => 3, 'Write' => 1],
'totalToolUseCount' => 4, // observed, not self-reported turn count
'productivityWarning' => null, // or advisory string (CJK-localised โ since v0.9.1)
'outputWarnings' => [], // since v0.9.1 โ filesystem audit findings
]
completed_empty โ zero tool calls observed. Re-dispatch or pick a stronger model.
completed + non-empty productivityWarning โ the child invoked tools but wrote no files (often fine for advisory consults; check the text).
Productivity instrumentation since v0.8.9. CJK localisation + filesystem audit since v0.9.1.
Output-directory audit + guard injection
Pass output_subdir to opt into both (a) a CJK-aware guard-block prepended to the child's prompt and (b) a post-exit filesystem scan:
$agent->run('...', [ 'output_subdir' => '/abs/path/to/reports/analyst-1', ]); // Audit catches: // - non-whitelisted extensions (defaults to .md / .csv / .png) // - consolidator-reserved filenames (summary.md / ๆ่ฆ.md / mindmap.md / ...) // - sibling-role sub-dirs (ceo / cfo / cto / marketing / ... or kebab-case role slugs) // Configurable via AgentOutputAuditor constructor. Never modifies disk.
Since v0.9.1
Provider-native tools
Any main brain can call these as regular tools โ no provider switch needed.
Moonshot server-hosted builtins (execute server-side; results inlined in the assistant reply):
| Tool | Attributes | Since |
|---|---|---|
KimiMoonshotWebSearchTool ($web_search) |
network | v0.9.0 |
KimiMoonshotWebFetchTool ($web_fetch) |
network | v0.9.1 |
KimiMoonshotCodeInterpreterTool ($code_interpreter) |
network, cost, sensitive | v0.9.1 |
Other provider-native tool families:
- Kimi โ
KimiFileExtractTool,KimiBatchTool,KimiSwarmTool,KimiMediaUploadTool - Qwen โ
QwenLongFileTool+dashscope_cache_controlfeature - GLM โ
glm_web_search,glm_web_reader,glm_ocr,glm_asr - MiniMax โ
minimax_tts,minimax_music,minimax_video,minimax_image
Agent Definitions (YAML / Markdown)
Auto-loaded from ~/.superagent/agents/ (user scope) and <project>/.superagent/agents/ (project scope). Three formats: .yaml, .yml, .md. Cross-format extend: inheritance.
# ~/.superagent/agents/reviewer.yaml name: reviewer description: Code reviewer with strict style enforcement extend: base-coder # can be .yaml / .yml / .md system_prompt: | You review PRs with a focus on correctness and hidden state. allowed_tools: [read, grep, glob] disallowed_tools: [write, edit, bash] model: claude-sonnet-4-5-20250929
<!-- ~/.superagent/agents/analyst.md --> --- name: analyst extend: reviewer model: gpt-5 --- Your job is to surface architectural risks. Write findings as Markdown.
Tool-list fields (allowed_tools, disallowed_tools, exclude_tools) accumulate through extend: chains. Cycle depth-limited.
Since v0.9.0
Skills
Markdown-based capabilities you can register globally and pull into any agent run:
superagent skills install ./my-skill.md
superagent skills list
superagent skills show review
superagent skills remove review
superagent skills path # show install directory
Skill markdown supports frontmatter with name, description, allowed_tools, system_prompt. Skill runs inherit the caller's provider.
MCP Integration
Server registration
superagent mcp list superagent mcp add sqlite stdio uvx --arg mcp-server-sqlite superagent mcp add brave stdio npx --arg @brave/mcp --env BRAVE_API_KEY=... superagent mcp remove sqlite superagent mcp status superagent mcp path
Config persists atomically at ~/.superagent/mcp.json.
OAuth-gated MCP servers
superagent mcp auth <name> # run RFC 8628 device flow superagent mcp reset-auth <name> # clear stored token superagent mcp test <name> # probe availability (stdio `command -v` or HTTP reachability)
Servers declaring an oauth: {client_id, device_endpoint, token_endpoint} block in their config use this flow. Since v0.9.0.
Declarative catalog + non-destructive sync
Drop a catalog at .mcp-servers/catalog.json (or .mcp-catalog.json) in your project root:
{
"mcpServers": {
"sqlite": {"command": "uvx", "args": ["mcp-server-sqlite"]},
"brave": {"command": "npx", "args": ["@brave/mcp"], "env": {"BRAVE_API_KEY": "k"}}
},
"domains": {
"baseline": ["sqlite"],
"all": ["sqlite", "brave"]
}
}
Sync to a project .mcp.json:
superagent mcp sync # full catalog superagent mcp sync --domain=baseline # only the "baseline" domain superagent mcp sync --servers=sqlite,brave # explicit subset superagent mcp sync --dry-run # preview, no disk writes
Non-destructive contract โ byte-equal disk hash โ unchanged; a user-edited file is kept as user-edited; first-time writes or our-last-hash matches become written. A manifest at <project>/.superagent/mcp-manifest.json tracks sha256 of every file we've written so stale entries clean up automatically.
Since v0.9.1
Wire Protocol
v1 โ line-delimited JSON (NDJSON), one event per line, self-describing via wire_version + type top-level fields. Foundation for IDE bridges, CI integrations, structured logs.
superagent --output json-stream "summarise src/" # Emits events like: # {"wire_version":1,"type":"turn.begin","turn_number":1} # {"wire_version":1,"type":"text.delta","delta":"I'll start by..."} # {"wire_version":1,"type":"tool.call","name":"read","input":{"path":"src/"}} # {"wire_version":1,"type":"turn.end","turn_number":1,"usage":{...}}
Transport (since v0.9.1)
Choose where the stream goes via a DSN:
| DSN | Meaning |
|---|---|
stdout (default) / stderr |
Standard streams |
file:///path/to/log.ndjson |
Append-mode file write |
tcp://host:port |
Connect to a listening TCP peer |
unix:///path/to/sock |
Connect to a listening unix socket |
listen://tcp/host:port |
Listen on TCP, accept one client |
listen://unix//path/to/sock |
Listen on unix socket, accept one client |
Programmatic use:
$factory = new SuperAgent\CLI\AgentFactory(); [$emitter, $transport] = $factory->makeWireEmitterForDsn('listen://unix//tmp/agent.sock'); // IDE plugin attaches, then: $agent->run($prompt, ['wire_emitter' => $emitter]); $transport->close();
Non-blocking peer socket means a dropped IDE doesn't stall the agent loop.
Wire Protocol v1 since v0.9.0. Socket / TCP / file transport since v0.9.1.
Retry, Errors & Observability
Layered retry
new Agent([ 'provider' => 'openai', 'request_max_retries' => 4, // HTTP connect / 4xx / 5xx (default 3) 'stream_max_retries' => 5, // reserved for mid-stream resume (Responses API) 'stream_idle_timeout_ms' => 60_000, // cURL low-speed cutoff on SSE (default 300 000) ]);
Jittered exponential backoff (0.9โ1.1ร multiplier) prevents thundering-herd retries from parallel workers. Retry-After header honoured exactly (no jitter โ the server knows best).
Since v0.9.1
Classified errors
Six subclasses of ProviderException emitted by OpenAIErrorClassifier against the response body's error.code / error.type / HTTP status:
try { $agent->run($prompt); } catch (\SuperAgent\Exceptions\Provider\ContextWindowExceededException $e) { // prompt was too long; compact history or swap models } catch (\SuperAgent\Exceptions\Provider\QuotaExceededException $e) { // monthly cap hit; notify operator } catch (\SuperAgent\Exceptions\Provider\UsageNotIncludedException $e) { // ChatGPT plan doesn't include this model; upgrade or switch to API key } catch (\SuperAgent\Exceptions\Provider\CyberPolicyException $e) { // policy rejection โ don't retry } catch (\SuperAgent\Exceptions\Provider\ServerOverloadedException $e) { // retryable with backoff; check $e->retryAfterSeconds } catch (\SuperAgent\Exceptions\Provider\InvalidPromptException $e) { // malformed body โ inspect and fix } catch (\SuperAgent\Exceptions\ProviderException $e) { // catch-all base; every subclass above extends this }
All subclasses extend ProviderException, so pre-existing catch (ProviderException) sites keep working unchanged.
Since v0.9.1
Health dashboard
superagent health # 5s cURL probe of every configured provider superagent health --all # include providers with no env key (useful for "what did I forget to set?") superagent health --json # machine-readable table; exits non-zero on any failure
Wraps ProviderRegistry::healthCheck() โ distinguishes auth rejection (401/403) from network timeout from "no API key" so an operator can fix the right thing without guessing.
Since v0.9.1
SSE parser hardening (since v0.9.0)
- Per-index tool-call assembly โ one streamed call split across N chunks now produces one tool-use block, not N fragments.
finish_reason: error_finishdetection โ DashScope-compat throttles raiseStreamContentError(retryable, HTTP 429) instead of silently polluting the message body.- Truncated tool-call JSON repair โ one-shot attempt to close unbalanced braces before falling back to an empty arg dict.
- Dual-shape cached-token reads โ
usage.prompt_tokens_details.cached_tokens(current OpenAI shape) ANDusage.cached_tokens(legacy) both populateUsage::cacheReadInputTokens.
Guardrails & Checkpoints
Loop detection (since v0.9.0)
Five detectors observe the streaming event bus; first trigger is sticky:
| Detector | Signal |
|---|---|
TOOL_LOOP |
Same tool + same normalised args 5ร in a row |
STAGNATION |
Same tool name 8ร regardless of args |
FILE_READ_LOOP |
โฅ 8 of last 15 tool calls are read-like, with cold-start exemption |
CONTENT_LOOP |
Same 50-char rolling window appears 10ร in streamed text |
THOUGHT_LOOP |
Same thinking-channel text appears 3ร |
new Agent([ 'provider' => 'openai', 'loop_detection' => true, // defaults // OR per-detector overrides: // 'loop_detection' => ['TOOL_LOOP' => 10, 'STAGNATION' => 15], ]);
Violations fan out as loop_detected wire events โ the agent keeps running, the host decides whether to intervene.
Checkpoints + shadow-git (since v0.9.0)
Every turn snapshots the agent state (messages, cost, usage). Attach a GitShadowStore and file-level snapshots land alongside in a separate bare git repo at ~/.superagent/history/<project-hash>/shadow.git โ never touches the user's own .git.
use SuperAgent\Checkpoint\CheckpointManager; use SuperAgent\Checkpoint\GitShadowStore; $mgr = new CheckpointManager(shadowStore: new GitShadowStore('/path/to/project')); $mgr->createCheckpoint($agentState, label: 'after-refactor'); // Later: $checkpoints = $mgr->list(); $mgr->restore($checkpoints[0]->id); $mgr->restoreFiles($checkpoints[0]); // plays back the shadow commit
Restore reverts tracked files and leaves untracked files in place for safety. The project's own .gitignore is respected (the shadow's worktree IS the project dir).
Permission modes
new Agent([ 'provider' => 'anthropic', 'permission_mode' => 'ask', // or 'default' / 'plan' / 'bypassPermissions' ]);
ask prompts the caller's PermissionCallbackInterface before any write-class tool. Wrap it in WireProjectingPermissionCallback to surface the request as a wire event for IDE prompts.
Standalone CLI
superagent # interactive REPL superagent "fix the login bug" # one-shot superagent init # initialize ~/.superagent/ superagent auth login <provider> # import OAuth login superagent auth status # show stored credentials superagent models list / update / refresh / status / reset superagent mcp list / add / remove / sync / auth / reset-auth / test / status / path superagent skills install / list / show / remove / path superagent swarm <prompt> # plan + execute a swarm superagent health [--all] [--json] [--providers=a,b,c] # provider reachability superagent eval [list|show|run] # capability evals โ ~/.superagent/model_scores.json superagent smart "<task>" # eval-score-driven plan + route + merge superagent smart "<task>" --max-cost 0.5 # abort if running spend exceeds the cap superagent smart show [<id|--last>] # list / inspect persisted runs superagent smart replay <id|--last> # re-execute a saved plan with new routing knobs
Options:
-m, --model <model> Model name
-p, --provider <provider> Provider key (openai, anthropic, openai-responses, ...)
--max-turns <n> Maximum agent turns (default 50)
-s, --system-prompt <prompt> Custom system prompt
--project <path> Project working directory
--json Output results as JSON
--output json-stream Emit NDJSON wire events
--verbose-thinking Show full thinking stream
--no-thinking Hide thinking
--plain Disable ANSI colours
--no-rich Legacy minimal renderer
-V, --version Show version
-h, --help Show help
Interactive commands (inside the REPL):
/help available commands
/model <name> switch model
/cost show cost tracking
/compact force context compaction
/session save|load|list|delete
/smart <task> eval-score-driven plan+route+merge (see `superagent smart`)
/workflows list/run dynamic workflows (get|plan|run|delete|create; --run|--plan)
/ultraplan <task> deep-plan a task into a runnable dynamic workflow
/ultrareview [target] multi-dimension review of the diff as a dynamic workflow
/deep-research <q> fan-out web research โ verify โ cited report as a dynamic workflow
/clear clear conversation
/quit exit
Standalone CLI since v0.8.6.
Laravel Integration
The service provider auto-registers when you composer require forgeomni/superagent:
// config/superagent.php return [ 'default_provider' => env('SUPERAGENT_PROVIDER', 'anthropic'), 'providers' => [ 'anthropic' => ['api_key' => env('ANTHROPIC_API_KEY')], 'openai' => ['api_key' => env('OPENAI_API_KEY')], 'openai-responses' => ['api_key' => env('OPENAI_API_KEY'), 'model' => 'gpt-5'], // ... ], 'agent' => [ 'max_turns' => 50, 'max_budget_usd' => 5.00, ], ];
use SuperAgent\Facades\SuperAgent; $result = SuperAgent::agent(['provider' => 'openai']) ->run('summarise this week\'s commits');
Artisan commands mirror the CLI:
php artisan superagent:chat "fix the bug"
php artisan superagent:mcp sync
php artisan superagent:models refresh
php artisan superagent:health --json
See docs/LARAVEL.md for queue integration, job dispatching, and the ai_usage_logs schema.
Host Integrations
Frameworks that embed SuperAgent โ typically multi-tenant platforms that store encrypted provider credentials in a database row and spin up an agent per request โ use ProviderRegistry::createForHost() instead of create(). The host passes a normalised shape and the SDK dispatches to the right constructor via per-provider adapters.
use SuperAgent\Providers\ProviderRegistry; // One call, every provider โ no `match ($type)` on the host side. $agent = ProviderRegistry::createForHost($sdkKey, [ 'api_key' => $aiProvider->decrypted_api_key, 'base_url' => $aiProvider->base_url, 'model' => $resolvedModel, 'max_tokens' => $extra['max_tokens'] ?? null, 'region' => $extra['region'] ?? null, 'credentials' => $extra, // opaque blob; adapter picks what it needs 'extra' => $extra, // provider-specific passthrough (organization, reasoning, verbosity, ...) ]);
Every ChatCompletions-style provider (Anthropic, OpenAI, OpenAI-Responses, OpenRouter, Ollama, LM Studio, Gemini, Kimi, Qwen, Qwen-native, GLM, MiniMax) uses the default pass-through adapter. Bedrock ships a built-in adapter that splits credentials.aws_access_key_id / aws_secret_access_key / aws_region into the AWS SDK's shape.
Plugins or hosts that need to customise an adapter register their own:
ProviderRegistry::registerHostConfigAdapter('my-custom-provider', function (array $host): array { return [ 'api_key' => $host['credentials']['my_custom_token'] ?? null, 'model' => $host['model'] ?? 'default-model', // ... arbitrary transform ]; });
New SDK provider keys in future releases register their own adapter (or ride the default one), so the host-side factory code never needs to grow a new match arm per release.
Since v0.9.2
Configuration reference
Every option accepted by the Agent constructor, grouped. Defaults in parentheses.
Provider selection
| Key | Accepts |
|---|---|
provider |
Registry key or an LLMProvider instance |
model |
Model id โ overrides provider default |
base_url |
URL โ overrides provider default; also triggers auto-detection (Azure) |
region |
intl / cn / us / hk / code (provider-specific) |
api_key |
Provider API key |
access_token + account_id |
OAuth (OpenAI ChatGPT / Anthropic Claude Code) |
auth_mode |
'api_key' (default) or 'oauth' |
organization |
OpenAI org id (adds OpenAI-Organization header) |
Agent loop
| Key | Default |
|---|---|
max_turns |
50 |
max_budget_usd |
0.0 (no cap) |
system_prompt |
null |
auto_mode |
false |
allowed_tools / denied_tools |
null / [] |
permission_mode |
'default' |
options |
[] (per-call defaults forwarded to provider) |
Per-call options ($agent->run($prompt, $options))
| Key | Since | Notes |
|---|---|---|
model / max_tokens / temperature / tool_choice / response_format |
v0.1.0 | Standard Chat Completions knobs |
features |
v0.8.8 | thinking / prompt_cache_key / dashscope_cache_control / ... routed via FeatureDispatcher |
extra_body |
v0.9.0 | Power-user escape hatch โ deep-merged into the request body |
loop_detection |
v0.9.0 | true (defaults), false, or threshold overrides |
idempotency_key |
v0.9.1 | Passthrough to AgentResult::$idempotencyKey |
reasoning |
v0.9.1 | Responses API โ {effort, summary} |
verbosity |
v0.9.1 | Responses API โ low / medium / high |
prompt_cache_key |
v0.9.0 | Cache key for Kimi + OpenAI Responses |
previous_response_id |
v0.9.1 | Responses API continuation |
store / include / service_tier / parallel_tool_calls |
v0.9.1 | Responses API |
client_metadata |
v0.9.1 | Responses API opaque key-value map |
trace_context / traceparent / tracestate |
v0.9.1 | W3C Trace Context injection |
output_subdir |
v0.9.1 | AgentTool guard-block + post-exit audit |
Retry + transport (provider-level)
| Key | Default | Since |
|---|---|---|
max_retries |
3 |
v0.1.0 (legacy single knob) |
request_max_retries |
3 (inherits max_retries) |
v0.9.1 |
stream_max_retries |
5 |
v0.9.1 |
stream_idle_timeout_ms |
300_000 |
v0.9.1 |
env_http_headers |
[] |
v0.9.1 |
http_headers |
[] |
v0.9.1 |
experimental_ws_transport |
false |
v0.9.1 (scaffold) |
azure_api_version |
'2025-04-01-preview' |
v0.9.1 (Azure only) |
Links
- CHANGELOG โ full per-release notes
- INSTALL โ install + first-run setup
- Advanced usage โ patterns, sample agents, debugging
- Native providers โ region maps + capability matrix
- Wire protocol โ v1 spec
- Features matrix โ which provider supports which feature
License
MIT โ see LICENSE.