Technical deep-dive into ClawRouter's internals.
┌─────────────────────────────────────────────────────────────┐
│ OpenClaw / Your App │
│ (OpenAI-compatible client) │
└─────────────────────────────────────────────────────────────┘
│
▼
┌─────────────────────────────────────────────────────────────┐
│ ClawRouter Proxy (localhost) │
│ ┌─────────────┐ ┌─────────────┐ ┌───────────────────┐ │
│ │ Dedup │→ │ Router │→ │ x402 Payment │ │
│ │ Cache │ │ (15-dim) │ │ (USDC on Base │ │
│ └─────────────┘ └─────────────┘ │ or Solana) │ │
│ └───────────────────┘ │
│ ┌─────────────┐ ┌─────────────┐ ┌───────────────────┐ │
│ │ Fallback │ │ Balance │ │ SSE Heartbeat │ │
│ │ Chain │ │ Monitor │ │ (streaming) │ │
│ │ │ │ (EVM/Solana)│ │ │ │
│ └─────────────┘ └─────────────┘ └───────────────────┘ │
└─────────────────────────────────────────────────────────────┘
│
┌─────────┴──────────┐
▼ ▼
┌────────────────────────┐ ┌────────────────────────────────┐
│ blockrun.ai/api │ │ sol.blockrun.ai/api │
│ (EVM / Base USDC) │ │ (Solana USDC) │
│ x402 EIP-712 signing │ │ x402 SVM signing │
└────────────────────────┘ └────────────────────────────────┘
│ │
└──────────────┬────────────────┘
▼
OpenAI / Anthropic / Google
Key Principles:
- 100% local routing — No API calls for model selection
- Client-side only — Your wallet key never leaves your machine
- Non-custodial — USDC stays in your wallet until spent
- Dual-chain — USDC on Base (EVM) or USDC on Solana; no SOL token accepted
POST /v1/chat/completions
{
"model": "blockrun/auto",
"messages": [{ "role": "user", "content": "What is 2+2?" }],
"stream": true
}
// SHA-256 hash of request body
const dedupKey = RequestDeduplicator.hash(body);
// Check completed cache (30s TTL)
const cached = deduplicator.getCached(dedupKey);
if (cached) {
return cached; // Replay cached response
}
// Check in-flight requests
const inflight = deduplicator.getInflight(dedupKey);
if (inflight) {
return await inflight; // Wait for original to complete
}// Extract user's last message
const prompt = messages.findLast((m) => m.role === "user")?.content;
// Run 14-dimension weighted scorer
const decision = route(prompt, systemPrompt, maxTokens, {
config: DEFAULT_ROUTING_CONFIG,
modelPricing,
});
// decision = {
// model: "google/gemini-2.5-flash",
// tier: "SIMPLE",
// confidence: 0.92,
// savings: 0.99,
// costEstimate: 0.0012,
// }const estimated = estimateAmount(modelId, bodyLength, maxTokens);
const sufficiency = await balanceMonitor.checkSufficient(estimated);
if (sufficiency.info.isEmpty) {
throw new EmptyWalletError(walletAddress);
}
if (!sufficiency.sufficient) {
throw new InsufficientFundsError({ ... });
}
if (sufficiency.info.isLow) {
onLowBalance({ balanceUSD, walletAddress });
}if (isStreaming) {
// Send 200 + headers immediately
res.writeHead(200, {
"content-type": "text/event-stream",
"cache-control": "no-cache",
});
// Heartbeat every 2s to prevent timeout
heartbeatInterval = setInterval(() => {
res.write(": heartbeat\n\n");
}, 2000);
}Base (EVM) — EIP-712 USDC:
1. Request → blockrun.ai/api
2. ← 402 Payment Required
{
"x402Version": 1,
"accepts": [{
"scheme": "exact",
"network": "base",
"maxAmountRequired": "5000", // $0.005 USDC
"resource": "https://blockrun.ai/api/v1/chat/completions",
"payTo": "0x..."
}]
}
3. Sign EIP-712 typed data (EIP-3009 TransferWithAuthorization) with EVM wallet key
4. Retry with X-PAYMENT header
5. ← 200 OK with response
Solana — SVM USDC:
1. Request → sol.blockrun.ai/api
2. ← 402 Payment Required
{
"x402Version": 1,
"accepts": [{
"scheme": "exact",
"network": "solana",
"maxAmountRequired": "5000", // $0.005 USDC (6 decimals)
"resource": "https://sol.blockrun.ai/api/v1/chat/completions",
"payTo": "<base58 address>"
}]
}
3. Build and sign Solana transaction (SPL Token USDC transfer) with Solana wallet key
- Wallet derived via SLIP-10 Ed25519 (BIP-44 m/44'/501'/0'/0', Phantom-compatible)
4. Retry with X-PAYMENT header (base64-encoded signed transaction)
5. ← 200 OK with response
Important: Both chains accept only USDC tokens. Sending SOL or ETH to the wallet will not fund API payments.
const FALLBACK_STATUS_CODES = [400, 401, 402, 403, 429, 500, 502, 503, 504];
for (const model of fallbackChain) {
const result = await tryModelRequest(model, ...);
if (result.success) {
return result.response;
}
if (result.isProviderError && !isLastAttempt) {
console.log(`Fallback: ${model} → next`);
continue;
}
break;
}// Convert non-streaming JSON to SSE format
// (BlockRun API returns JSON, we simulate SSE)
// Chunk 1: role
data: {"id":"...","object":"chat.completion.chunk","choices":[{"delta":{"role":"assistant"}}]}
// Chunk 2: content
data: {"id":"...","object":"chat.completion.chunk","choices":[{"delta":{"content":"4"}}]}
// Chunk 3: finish
data: {"id":"...","object":"chat.completion.chunk","choices":[{"delta":{},"finish_reason":"stop"}]}
data: [DONE]The routing engine uses a 15-dimension weighted scorer that runs entirely locally:
function classifyByRules(
prompt: string,
systemPrompt: string | undefined,
tokenCount: number,
config: ScoringConfig,
): ClassificationResult {
let score = 0;
const signals: string[] = [];
// Dimension 1: Reasoning markers (weight: 0.18)
const reasoningCount = countKeywords(prompt, config.reasoningKeywords);
if (reasoningCount >= 2) {
score += 0.18 * 2; // Double weight for multiple markers
signals.push("reasoning");
}
// Dimension 2: Code presence (weight: 0.15)
if (hasCodeBlock(prompt) || countKeywords(prompt, config.codeKeywords) > 0) {
score += 0.15;
signals.push("code");
}
// ... 13 more dimensions
// Sigmoid calibration
const confidence = sigmoid(score, (k = 8), (midpoint = 0.5));
return { score, confidence, tier: selectTier(score, confidence), signals };
}function selectTier(score: number, confidence: number): Tier | null {
// Special case: 2+ reasoning markers → REASONING at high confidence
if (signals.includes("reasoning") && reasoningCount >= 2) {
return "REASONING";
}
if (confidence < 0.7) {
return null; // Ambiguous → default to MEDIUM
}
if (score < 0.3) return "SIMPLE";
if (score < 0.6) return "MEDIUM";
if (score < 0.8) return "COMPLEX";
return "REASONING";
}Certain conditions force tier assignment:
// Large context → COMPLEX
if (tokenCount > 100000) {
return { tier: "COMPLEX", method: "override:large_context" };
}
// Structured output (JSON/YAML) → min MEDIUM
if (systemPrompt?.includes("json") || systemPrompt?.includes("yaml")) {
return { tier: Math.max(tier, "MEDIUM"), method: "override:structured" };
}ClawRouter uses the x402 protocol for micropayments. Both chains use the same flow; the signing step differs:
┌────────────┐ ┌──────────────────────┐ ┌────────────┐
│ Client │────▶│ BlockRun API │────▶│ Provider │
│ (ClawRouter) │ (Base: blockrun.ai │ │ (OpenAI) │
└────────────┘ │ Sol: sol.blockrun) │ └────────────┘
│ │
│ 1. Request │
│─────────────────▶│
│ │
│ 2. 402 + price │
│◀─────────────────│
│ │
│ 3. Sign payment │
│ Base: EIP-712 │
│ Solana: SVM tx │
│ (USDC only) │
│ │
│ 4. Retry + sig │
│─────────────────▶│
│ │
│ 5. Response │
│◀─────────────────│
const typedData = {
types: {
TransferWithAuthorization: [
{ name: "from", type: "address" },
{ name: "to", type: "address" },
{ name: "value", type: "uint256" },
{ name: "validAfter", type: "uint256" },
{ name: "validBefore", type: "uint256" },
{ name: "nonce", type: "bytes32" },
],
},
primaryType: "TransferWithAuthorization",
domain: { name: "USD Coin", version: "2", chainId: 8453, verifyingContract: USDC_BASE },
message: {
from: walletAddress,
to: payTo,
value: BigInt(5000), // 0.005 USDC (6 decimals)
validAfter: BigInt(0),
validBefore: BigInt(Math.floor(Date.now() / 1000) + 3600),
nonce: crypto.getRandomValues(new Uint8Array(32)),
},
};
const signature = await account.signTypedData(typedData);// Wallet derived via SLIP-10 Ed25519 — Phantom-compatible
// Path: m/44'/501'/0'/0'
const solanaAccount = await deriveSlip10Ed25519Key(mnemonic, "m/44'/501'/0'/0'");
// Build SPL Token USDC transfer instruction
const transaction = buildSolanaPaymentTransaction({
from: solanaAddress,
to: payTo, // base58 recipient
mint: USDC_SOLANA, // EPjFWdd5AufqSSqeM2qN1xzybapC8G4wEGGkZwyTDt1v
amount: BigInt(5000), // 0.005 USDC (6 decimals)
});
const signedTx = await signTransaction(transaction, solanaAccount);
// Encoded as base64 in X-PAYMENT headerTo skip the 402 round trip:
// Estimate cost before request
const estimated = estimateAmount(modelId, bodyLength, maxTokens);
// Pre-sign payment with estimate (+ 20% buffer)
const preAuth: PreAuthParams = { estimatedAmount: estimated };
// Request with pre-signed payment
const response = await payFetch(url, init, preAuth);Prevents double-charging when clients retry after timeout:
class RequestDeduplicator {
private cache = new Map<string, CachedResponse>();
private inflight = new Map<string, Promise<CachedResponse>>();
private TTL_MS = 30_000;
static hash(body: Buffer): string {
return createHash("sha256").update(body).digest("hex");
}
getCached(key: string): CachedResponse | undefined {
const entry = this.cache.get(key);
if (entry && Date.now() - entry.completedAt < this.TTL_MS) {
return entry;
}
return undefined;
}
}Prevents upstream timeout while waiting for x402 payment:
0s: Request received
0s: → 200 OK, Content-Type: text/event-stream
0s: → : heartbeat
2s: → : heartbeat (client stays connected)
4s: → : heartbeat
5s: x402 payment completes
5s: → data: {"choices":[...]}
5s: → data: [DONE]
Avoids RPC calls on every request. Dual-chain monitors are chain-aware:
// EVM monitor (Base): reads USDC balance via eth_call on Base RPC
class BalanceMonitor {
private cachedBalance: bigint | undefined;
private cacheTime = 0;
private CACHE_TTL_MS = 60_000; // 1 minute
async checkBalance(): Promise<BalanceInfo> {
if (this.cachedBalance !== undefined && Date.now() - this.cacheTime < this.CACHE_TTL_MS) {
return this.formatBalance(this.cachedBalance);
}
// Fetch USDC balance from Base RPC
const balance = await this.fetchUSDCBalance(); // ERC-20 balanceOf call
this.cachedBalance = balance;
this.cacheTime = Date.now();
return this.formatBalance(balance);
}
deductEstimated(amount: bigint): void {
if (this.cachedBalance !== undefined) {
this.cachedBalance -= amount;
}
}
}
// Solana monitor: reads SPL Token USDC balance via getTokenAccountBalance
class SolanaBalanceMonitor {
// Same interface as BalanceMonitor — proxy.ts uses AnyBalanceMonitor union type
// Retries once on empty to handle flaky public RPC endpoints
// Cache TTL 60s; startup balance never cached (forces fresh read after install)
}
// proxy.ts selects the correct monitor at startup:
const balanceMonitor: AnyBalanceMonitor =
paymentChain === "solana"
? new SolanaBalanceMonitor(solanaAddress, rpcUrl)
: new BalanceMonitor(evmAddress, rpcUrl);Detects and reuses existing proxy to avoid EADDRINUSE:
async function startProxy(options: ProxyOptions): Promise<ProxyHandle> {
const port = options.port ?? getProxyPort();
// Check if proxy already running
const existingWallet = await checkExistingProxy(port);
if (existingWallet) {
// Return handle that uses existing proxy
return {
port,
baseUrl: `http://127.0.0.1:${port}`,
walletAddress: existingWallet,
close: async () => {}, // No-op
};
}
// Start new proxy
const server = createServer(...);
server.listen(port, "127.0.0.1");
// ...
}src/
├── index.ts # Plugin entry, OpenClaw integration
├── proxy.ts # HTTP proxy server, request handling, chain selection
├── provider.ts # OpenClaw provider registration
├── models.ts # 41+ model definitions with pricing
├── auth.ts # Wallet key resolution (file → env → generate)
├── wallet.ts # BIP-39 mnemonic, EVM + Solana key derivation (SLIP-10)
├── x402.ts # EVM EIP-712 payment signing, @x402/fetch
├── balance.ts # EVM USDC balance monitoring (Base RPC)
├── solana-balance.ts # Solana USDC balance monitoring (SPL Token)
├── payment-preauth.ts # Pre-authorization caching (EVM only)
├── dedup.ts # Request deduplication (SHA-256 → cache)
├── logger.ts # JSON usage logging to disk
├── errors.ts # Custom error types
├── retry.ts # Fetch retry with exponential backoff
├── version.ts # Version from package.json
└── router/
├── index.ts # route() entry point
├── rules.ts # 15-dimension weighted scorer (9-language)
├── selector.ts # Tier → model selection + fallback
├── config.ts # Default routing configuration (ECO/AUTO/PREMIUM/AGENTIC)
└── types.ts # TypeScript type definitions
| File | Purpose |
|---|---|
proxy.ts |
Core request handling, SSE simulation, fallback chain |
wallet.ts |
BIP-39 mnemonic generation, EVM + Solana (SLIP-10) derivation |
router/rules.ts |
15-dimension weighted scorer, 9-language keyword sets |
x402.ts |
EIP-712 typed data signing, payment header formatting |
balance.ts |
USDC balance via Base RPC (EVM), caching, thresholds |
solana-balance.ts |
USDC balance via Solana RPC (SPL Token), caching, retries |
payment-preauth.ts |
Pre-authorization cache (EVM; skipped for Solana) |
dedup.ts |
SHA-256 hashing, 30s response cache |