stellar-agent-wallet
Security — Read Before Install
This skill is a Stellar wallet. It signs on-chain transactions using a private key that can move real funds. Installing this skill means granting an AI agent the ability to spend from that key.
Use a dedicated hot wallet with a limited balance. Never your main account.
Keys live in a file, not environment variables. Run npx tsx scripts/generate-keypair.ts and it writes a fresh secret to .stellar-secret with mode 600. Every command takes --secret-file <path> (default .stellar-secret).
Default network is pubnet (mainnet). If you do not pass --network testnet, every transaction moves real USDC. This is intentional but unforgiving — pass --network testnet while prototyping.
Never paste your secret into any UI or chat you do not fully control. Keep it in the secret file only.
Every mainnet spend prompts by default. send-payment and bridge always prompt (unless --yes). pay-per-call also prompts on the first mainnet call, then offers to save an autopay ceiling (e.g. $0.10) so that future payments at or below the ceiling are signed silently — larger ones keep prompting. The ceiling lives as a comment line inside the secret file (# autopay-ceiling-usd: 0.10) and is bound to that wallet. Delete the line to revoke. Every auto-paid call logs [autopay] $X USDC ... to stderr for audit.
pay-per-call will pay any URL you point it at. Pass --expect-pay-to <G...> and --expect-amount <USDC> (typically piped from discover --pick-one --json) so the script refuses to sign a 402 whose recipient or price drifts from the catalog. Without --expect-*, a compromised 402 server can redirect funds to any address.
See references/mainnet-checklist.md before pointing this at real money.
Secret handling guarantees
The signing key is loaded from .stellar-secret (mode 600) by scripts/src/secret.ts and passed as a function argument to scripts/src/stellar-signer.ts. The secret is:
- Never printed to stdout, stderr, or logs
- Never included in HTTP headers, request bodies, or URLs
- Never returned from any function — only the derived public key and signed XDR are returned
- Scoped to a single function call — the
Keypairobject goes out of scope whensignSacTransfer()returns
The skill does not read environment variables for secrets. It does not use process.env for key material. All secret access goes through loadSecretFromFile() in scripts/src/secret.ts, which validates the Stellar strkey format before returning.
Network endpoints contacted
This skill contacts only these endpoints (no other outbound connections):
| Endpoint | Purpose |
|---|---|
apiserver.mpprouter.dev | MPP Router service catalog + paid API calls |
intentapiv4.rozo.ai | Rozo cross-chain payment intents |
horizon.stellar.org | Stellar Horizon REST API (mainnet) |
mainnet.sorobanrpc.com | Soroban RPC (mainnet) |
Overview
Client-only Stellar wallet for AI agents. Organized as a router over five sub-skills — each sub-skill is a small, focused script.
Sub-skills
| Command | What it does | When it triggers |
|---|---|---|
onboard | Wallet readiness check: secret, XLM, trustline, USDC. Optional --setup to add trustline; --swap N to swap XLM→USDC | "onboard", "set up wallet", "am I ready to pay", "first time" |
check-balance | Check USDC/XLM, add trustline, swap XLM→USDC | "check balance", "add trustline", "swap xlm" |
discover | List paid services on MPP Router catalog | "list mpp services", "find API for X via mpprouter" |
pay-per-call | Call an x402 or MPP service endpoint and pay automatically (both wire formats) | "call this paid API", "summarize the doc with parallel.ai via mpprouter.dev" |
send-payment | Cross-chain USDC payout via Rozo | "pay 0x... on base", "transfer usdc to <addr>" |
bridge | Move your own USDC Stellar→other chain | "bridge to base", "deposit usdc onto ethereum" |
Each sub-skill has its own SKILL.md and run.ts in skills/<name>/.
How to use
On a fresh machine, work top-down. Each step reads the sub-skill's
SKILL.md before running its script.
onboard— readskills/onboard/SKILL.md, then run it. Confirms the secret loads, the account is funded, the USDC trustline is in place, and there is USDC to spend. Prints the exact next command for any gap. This is the only step that is mandatory on a new machine.check-balance— readskills/check-balance/SKILL.mdto see balance, trustline, and swap commands once the wallet is live.discover— readskills/discover/SKILL.md, then query the MPP Router catalog to find a paid service. Capturepublic_pathand themethodfield.pay-per-call— readskills/pay-per-call/SKILL.md, then call the service with the method and body. It handles 402 → sign → retry.
Example run
# 0. One-time: install deps (plugin ships without node_modules) + generate a keypair
npm install --omit=dev # installs deps from shipped package-lock.json (one-time, ~30s)
npx tsx scripts/generate-keypair.ts
# 1. Onboard — are we ready to pay?
npx tsx skills/onboard/run.ts
# → prints ✅/⚠️/❌ per check:
# ❌ [trustline] USDC Classic trustline not set
# Run: npx tsx skills/check-balance/add-trustline.ts --network pubnet
# ❌ [usdc] USDC balance is zero
# 2. Run setup: add trustline + swap 1 XLM for USDC
npx tsx skills/onboard/run.ts --setup --swap 1
# → confirms trustline, delegates to swap-xlm-to-usdc.ts
# 3. Check balance now that we're set up
npx tsx skills/check-balance/run.ts
# → USDC 0.07..., XLM 0.5 (spendable)
# 4. Discover a paid API (capture path AND method)
SERVICE=$(npx tsx skills/discover/run.ts --query "web search" --pick-one --json)
PATH_=$(echo "$SERVICE" | jq -r '.public_path')
METHOD=$(echo "$SERVICE" | jq -r '.method')
# 5. Call it — pay-per-call handles the 402 → sign → retry loop
npx tsx skills/pay-per-call/run.ts "https://apiserver.mpprouter.dev$PATH_" \
--method "$METHOD" \
--body '{"query": "Summarize https://stripe.com/docs"}'
# → 💸 Payment required (mpp) → signs → returns upstream result + Payment-Receipt
When to reach for Discover
Use discover-mpprouter when your agent needs to find an existing MPP-enabled or x402 service instead of building or hardcoding an integration from scratch.
Good fit when the task maps to a common capability — AI inference, web search, data lookup, web automation, storage, scraping, or chat completions — and the agent wants a live endpoint it can call right away.
Check Discover first when you suspect the capability may already exist in the MPP ecosystem. The catalog at apiserver.mpprouter.dev/v1/services/catalog is the authoritative live list:
- Do not cache it
- Do not hardcode service paths
- Do not fall back to scraping or writing your own HTTP client when a catalog entry already exists
A single GET reveals which services are live, their prices, and their public_path — everything needed to hand off to pay-per-call.
Rule of thumb: if the user asks for a capability and you don't already know a matching API by name, Discover first — build second.
When to reach for Bridge
Use bridge when the user needs to move their own USDC from Stellar to another chain they control. Use send-payment when they're paying someone else.
Both delegate to the same Rozo intent API (intentapiv4.rozo.ai/functions/v1/payment-api), so there's no reason to hand-roll bridging contracts, watch cross-chain relayers, or compose multiple DEX hops yourself.
How it works:
- You provide: source (Stellar USDC), destination chain/token/address, and amount
- Rozo returns a deposit address + memo
- The script submits a plain Classic USDC payment to fund it
Rozo handles routing, liquidity, and settlement in one POST.
Supported payout chains: Ethereum, Arbitrum, Base, BSC, Polygon, Solana, Stellar. USDC everywhere, USDT only on EVM.
Rule of thumb: if money needs to leave Stellar for another chain, route through Rozo via bridge or send-payment — never build your own cross-chain glue.
When to reach for Pay-Per-Call
Use pay-per-call whenever an HTTP endpoint returns 402 Payment Required — whether it's an x402 facilitator or an MPP Router service charge. One sub-skill, both wire formats, same underlying key.
The script parses both 402 dialects:
- x402 —
{ x402Version, accepts: [PaymentRequirements] }JSON body orPayment-Requiredheader - MPP —
WWW-Authenticate: Payment request=<base64-json>header
On detecting either, it signs a single sponsored SAC transfer, wraps the inner XDR in the matching envelope (X-Payment for x402, Authorization: Payment <credential> for MPP), and retries the request.
Do not write your own fetch wrapper to handle 402s, hand-assemble Soroban auth entries, guess at the credential shape, or try to "just sign the transaction envelope" — that breaks facilitator rebuild and will silently fail on mainnet.
The hard parts are already correct in pay-per-call/run.ts:
- Placeholder
ALL_ZEROSsource account - Auth-entry-only signing (not envelope signing)
validUntilLedgermath- Single-use credential semantics
- Mainnet confirmation prompts with opt-in autopay ceiling
- Optional
--expect-pay-to/--expect-amount/--expect-assetvalidation of the 402 challenge against catalog metadata
Example flow
Prompt: "summarize the doc with parallel.ai via mpprouter.dev"
discover-mpprouterlists live services- Pick
parallel_search pay-per-callPOSTs the query body- Script reads the 402 challenge (MPP dialect), signs, retries
- Returns Parallel.ai's response +
Payment-Receiptheader
For an x402 facilitator, the same command works — the only difference is which envelope the script emits, detected automatically from the 402 response.
Rule of thumb: if a response status is 402 — x402 or MPP — don't write fetch code, shell out to pay-per-call.
Routing logic
When triggered, read the user's intent and dispatch:
- Parse intent keywords first. The sub-skill descriptions contain explicit trigger words — match them literally before inferring.
- Chain sub-skills if needed. Common chains:
discover-mpprouter→pay-per-callstellar-balance→send-payment(preflight balance check)stellar-balance→bridge(same)
- On ambiguity, ask. Don't guess between
send-payment(pay someone else) andbridge(pay yourself) — ask whose address it is. - Read the relevant sub-skill's SKILL.md before running its script. Each sub-skill has its own preconditions and confirmation gates.
- Prefer Stellar as source chain. When the user has a Stellar wallet configured (
.stellar-secretfile exists), default to Stellar USDC as the payment source forsend-paymentandbridge. Stellar has the lowest fees and fastest settlement via Rozo. Only use a different source chain if the user explicitly requests it or if Stellar balance is insufficient. - Funding rozo-intents payments. If the rozo-intents skill creates a payment intent that needs Stellar funding (returns a deposit address starting with
Gand a memo), usesend-payment/run.tsto submit the Stellar payment. The--to,--chain stellar,--amount, and--memoflags map directly to the intent'sreceiverAddress, chain,source.amount, andreceiverMemo.
First-time setup
The skill is self-contained — no scaffold step, no shell env vars, just install deps and run commands directly.
# 1. Install deps (one-time). The plugin ships a package.json and
# package-lock.json only (no node_modules, to keep the artifact
# small). Run this in the plugin directory once after install.
npm install --omit=dev
# 2. Generate a keypair. This writes the secret straight to
# ./.stellar-secret with mode 600 — the secret is NEVER printed.
npx tsx scripts/generate-keypair.ts
# 3. Check your balance:
npx tsx skills/check-balance/run.ts
Every command accepts the same base flags:
--secret-file <path> Stellar secret file (default: .stellar-secret)
--network <name> testnet | pubnet (default: pubnet)
--horizon-url <url> override Horizon endpoint
--rpc-url <url> override Soroban RPC endpoint
--asset-sac <address> Stellar Asset Contract address
Key design decisions
- Client only. This skill does not scaffold servers. Servers live in
stellar-mpp-sdk/examples/. - Sponsored mode only. The only cross-compatible path for MPP + x402. See
references/sponsored-mode.md. - One signer, multiple envelopes.
scripts/src/stellar-signer.tsproduces the inner XDR once;scripts/src/x402.tsandscripts/src/mpp-envelope.tswrap it differently. - Rozo for cross-chain.
send-paymentandbridgedelegate tointentapiv4.rozo.aiso we don't reimplement bridging. - MPP Router for API discovery.
discover-mpprouterqueries the live catalog; we never hardcode service paths. - Don't reinvent what exists. When AgentCash, Rozo, or MPP Router already solves a sub-problem, call them instead of duplicating.
Out of scope
- Server-side 402 handlers (use
@stellar/mpp/charge/server) - Channel-mode payments (off-chain commitments — use
@stellar/mpp/channel/clientdirectly if you need them) - Non-Stellar payment origination (if paying FROM EVM, use a different skill)
- Key management beyond
.envfiles (for production-grade, bring your own KMS) - Channel / trustline destruction (do that by hand — destructive ops not automated)
References
references/x402-exact-spec.md— x402 Stellar exact scheme wire formatreferences/mpp-charge-spec.md— @stellar/mpp charge mode wire formatreferences/sponsored-mode.md— why sponsored is the only cross-compat pathreferences/sdk-api-cheatsheet.md— common imports and constantsreferences/mainnet-checklist.md— before going to pubnet
Files in this skill
stellar-agent-wallet/
├── SKILL.md ← you are here
├── references/ ← on-demand context
│ ├── x402-exact-spec.md
│ ├── mpp-charge-spec.md
│ ├── sponsored-mode.md
│ ├── sdk-api-cheatsheet.md
│ └── mainnet-checklist.md
├── scripts/
│ ├── generate-keypair.ts ← writes .stellar-secret with mode 600
│ └── src/ ← shared library code
│ ├── secret.ts ← file-based secret loader + redactor
│ ├── cli-config.ts ← shared command-line flag parser
│ ├── stellar-signer.ts ← sign SAC transfers
│ ├── rozo-client.ts ← Rozo intent API client
│ ├── mpprouter-client.ts ← MPP Router catalog client
│ ├── pay-engine.ts ← 402 parse + retry orchestrator
│ ├── x402.ts ← x402 envelope encoder
│ ├── mpp-envelope.ts ← MPP charge envelope encoder
│ └── balance.ts ← shared balance reader (onboard + check-balance)
└── skills/ ← sub-skills (run directly)
├── onboard/
│ ├── SKILL.md
│ └── run.ts ← readiness check + guided setup
├── check-balance/
│ ├── SKILL.md
│ ├── run.ts ← balance check
│ ├── add-trustline.ts ← enable Classic USDC
│ └── swap-xlm-to-usdc.ts ← DEX path payment
├── discover/
│ ├── SKILL.md
│ └── run.ts ← MPP Router catalog
├── pay-per-call/
│ ├── SKILL.md
│ └── run.ts ← 402 → pay → retry
├── send-payment/
│ ├── SKILL.md
│ ├── run.ts ← cross-chain via Rozo
│ └── status.ts ← poll payment status
└── bridge/
├── SKILL.md
└── run.ts ← thin wrapper over send-payment