A solid implementation for trading on Hyperliquid's on-chain DEX that handles both crypto and stock perpetuals plus spot tokens. The skill automatically manages the messy bits like checking cross-margin balances, resolving RWA tickers (NVIDIA becomes xyz:NVDA), setting leverage before orders, and verifying fills after execution. It covers the full trading lifecycle from USDC deposits to stop losses, with separate tools for perps versus spot. The automatic balance checking and fill verification means you can just say "buy $50 of Bitcoin with 10x" and it handles the sequence properly. Requires wallet policy setup but then trading flows are straightforward.
npx -y skills add starchild-ai-agent/official-skills --skill hyperliquid --agent claude-codeInstalls into .claude/skills of the current project.
Trade perpetual futures and spot tokens on Hyperliquid, a fully on-chain decentralized exchange. Orders are signed using this agent's EVM wallet and submitted directly to the Hyperliquid L1.
Before trading, the wallet policy must be active. Load the wallet-policy skill and propose the standard wildcard policy (deny key export + allow *). This covers all Hyperliquid operations — USDC deposits, EIP-712 order signing, and withdrawals.
Hyperliquid has two execution modes. Use the right one for your workflow:
Use hl_* tools directly in agent conversations and task scripts that run inside the Starchild tool runtime.
check → execute → verify)hl_* tools are tool-runtime capabilities, not normal Python importsFor standalone services (FastAPI bots, daemons, web backends), call Hyperliquid directly via the bundled client:
skills/hyperliquid/client.py (HyperliquidClient)hl_* tools are not importable as from ... import hl_order in plain Python servicesHyperliquidClient for order, cancel, open orders, fills, account/start, /stop, /status, /history) around that statefrom skills.hyperliquid.client import HyperliquidClient
client = HyperliquidClient()
address = await client._get_address() # wallet address used for read endpoints
# Query
account = await client.get_account_state(address)
opens = await client.get_open_orders(address)
fills = await client.get_user_fills(address)
# Place order (example)
res = await client.place_order(
coin="BTC",
is_buy=True,
size=0.001,
price=95000,
order_type="limit",
)
# Cancel all BTC orders
await client.cancel_all("BTC")
| Tool | What it does |
|---|---|
hl_total_balance | Check how much you can trade with (use this for balance checks!) |
hl_account | See your open positions and PnL |
hl_balances | See your token holdings (USDC, HYPE, etc.) |
hl_market | Get current prices for crypto or stocks |
hl_orderbook | Check order book depth and liquidity |
hl_fills | See recent trade fills and execution prices |
hl_candles | Get price charts (1m, 5m, 1h, 4h, 1d) |
hl_funding | Check funding rates for perps |
hl_open_orders | See pending orders |
| Tool | What it does |
|---|---|
hl_order | Buy or sell perps (crypto/stocks) |
hl_spot_order | Buy or sell spot tokens |
hl_tpsl_order | Place stop loss or take profit orders |
hl_leverage | Set leverage (1x to asset max) |
hl_cancel | Cancel a specific order |
hl_cancel_all | Cancel all open orders |
hl_modify | Change order price or size |
| Tool | What it does |
|---|---|
hl_deposit | Add USDC from Arbitrum (min $5) |
hl_withdraw | Send USDC to Arbitrum (1 USDC fee, ~5 min) |
hl_transfer_usd | Move USDC between spot/perp (rarely needed) |
Just tell the agent what you want to trade - it handles everything automatically.
Examples:
User: "Buy $20 of Bitcoin with 5x leverage"
Agent: [checks balance → sets leverage → places order → reports fill]
Result: "✓ Bought 0.0002 BTC at $95,432 with 5x leverage. Position opened."
User: "Long NVIDIA with $50, 10x"
Agent: [auto-detects NVIDIA = xyz:NVDA → executes → verifies]
Result: "✓ Bought 0.25 NVDA at $198.50 with 10x leverage. Filled at $198.62."
User: "Sell my ETH position"
Agent: [checks position size → closes → reports PnL]
Result: "✓ Sold 0.5 ETH at $3,421. Realized PnL: +$12.50"
You don't need to:
Just say what you want, the agent handles the rest.
🤖 As the agent, you should ALWAYS do these automatically (never ask the user):
hl_total_balance before EVERY trade to see total available marginhl_leverage before placing orders (unless user specifies not to)hl_fills to check if it filled🎯 User just says: "buy X" or "sell Y" or "long Z with $N"
🔧 You figure out:
📊 Balance checking hierarchy:
hl_total_balance - shows ACTUAL available margin regardless of account modehl_account for balance - may show $0 even if funds availablehl_balances for margin - only shows spot tokens🚀 Be proactive, not reactive:
hl_account() # Default crypto perps account
hl_account(dex="xyz") # Builder dex (RWA/stock perps) account
Returns marginSummary (accountValue, totalMarginUsed, withdrawable) and assetPositions array with each position's coin, szi (signed size), entryPx, unrealizedPnl, leverage.
Important: Builder perps (xyz:NVDA, xyz:TSLA, etc.) have separate clearinghouses. Always check the correct dex when trading RWA/stock perps.
hl_balances()
Returns balances array with coin, hold, total for USDC and all spot tokens.
hl_market() # All mid prices
hl_market(coin="BTC") # BTC price + metadata (maxLeverage, szDecimals)
All order tools (hl_order, hl_spot_order, hl_tpsl_order, hl_modify)
use the same side parameter. Use "buy" or "sell" — these are the
documented values and should be your default.
For safety, the tools also accept these aliases so a model guess doesn't reverse direction on a leveraged order:
"buy", "B", "bid", "long", "L", 1, true"sell", "S", "A", "ask", "short", 0, falseAn unrecognized value will fail the call with a clear error — the tool
never defaults to sell (or buy) when side is ambiguous. This is intentional:
silently reversing direction on a leveraged position is the worst failure mode.
Note: Hyperliquid's L1 wire protocol uses "B" and "A" internally, but the
tool interface here is buy/sell. Stick to buy/sell in your calls and
you will never be surprised.
hl_order(coin="BTC", side="buy", size=0.01, price=95000)
Places a GTC limit buy for 0.01 BTC at $95,000.
hl_order(coin="ETH", side="sell", size=0.1)
Omitting price submits an IoC order at mid price +/- 3% slippage.
Parameter format behavior:
size as number, reduce_only as boolean)"0.01" → 0.01"true"/"false" → true/false"5"/"5.0" → 5hl_order(coin="BTC", side="buy", size=0.01, price=94000, order_type="alo")
ALO (Add Liquidity Only) = post-only. Rejected if it would immediately fill.
Practical guardrail for bots: If your ALO price is too close to mid (often within ~0.1% on liquid pairs), Hyperliquid may reject it. For market-making/grid bots, compute current mid first and skip or shift levels that sit inside your no-cross buffer zone.
hl_tpsl_order(coin="BTC", side="sell", size=0.01, trigger_px=90000, tpsl="sl")
Automatically sells 0.01 BTC if the price drops to $90,000. Executes as market order when triggered.
For a limit order when triggered (instead of market):
hl_tpsl_order(coin="BTC", side="sell", size=0.01, trigger_px=90000, tpsl="sl", is_market=false, limit_px=89900)
hl_tpsl_order(coin="ETH", side="sell", size=0.5, trigger_px=3500, tpsl="tp")
Automatically sells 0.5 ETH if the price rises to $3,500. Executes as market order when triggered.
hl_order(coin="BTC", side="sell", size=0.01, reduce_only=true)
Use reduce_only=true to ensure it only closes, never opens a new position.
hl_spot_order(coin="HYPE", side="buy", size=10, price=25.0)
Spot orders use the same interface — just specify the token name.
hl_cancel(coin="BTC", order_id=12345678)
Get order_id from hl_open_orders.
hl_cancel_all() # Cancel everything
hl_cancel_all(coin="BTC") # Cancel only BTC orders
hl_modify(order_id=12345678, coin="BTC", side="buy", size=0.02, price=94500)
hl_leverage(coin="BTC", leverage=10) # 10x cross margin
hl_leverage(coin="ETH", leverage=5, cross=false) # 5x isolated margin
hl_transfer_usd(amount=1000, to_perp=true) # Spot → Perp
hl_transfer_usd(amount=500, to_perp=false) # Perp → Spot
Note: Usually not needed - funds are automatically shared. Only use if you get an error saying you need to transfer.
hl_withdraw(amount=100) # Withdraw to own wallet
hl_withdraw(amount=50, destination="0xABC...") # Withdraw to specific address
Fee: 1 USDC deducted by Hyperliquid. Processing takes ~5 minutes.
hl_deposit(amount=500)
Sends USDC from the agent's Arbitrum wallet to the Hyperliquid bridge contract. Minimum deposit: 5 USDC. Requires USDC balance on Arbitrum.
hl_candles(coin="BTC", interval="1h", lookback=48)
Intervals: 1m, 5m, 15m, 1h, 4h, 1d. Lookback in hours.
hl_funding() # All predicted fundings
hl_funding(coin="BTC") # BTC predicted + 24h history
hl_fills(limit=10)
When a user asks to trade a ticker, you need to determine whether it's a native crypto perp (use plain name) or an RWA/stock perp (use xyz:TICKER prefix).
"BTC", "ETH", "SOL", "DOGE", "HYPE", etc.xyz: prefix: "xyz:NVDA", "xyz:TSLA", "xyz:GOLD", etc.hl_market(coin="X") — if it returns a price, it's a crypto perphl_market(dex="xyz") to list all RWA markets and search the resultsxyz: prefix)| Category | Examples |
|---|---|
| US Stocks | xyz:NVDA, xyz:TSLA, xyz:AAPL, xyz:MSFT, xyz:AMZN, xyz:GOOG, xyz:META, xyz:TSM |
| Commodities — Metals | xyz:GOLD, xyz:SILVER, xyz:COPPER, xyz:PLATINUM, xyz:PALLADIUM, xyz:ALUMINIUM |
| Commodities — Energy | xyz:CL (WTI), xyz:BRENTOIL, xyz:NATGAS, xyz:TTF (EU Gas) |
| Commodities — Agriculture | xyz:CORN, xyz:WHEAT |
| Commodities — Other | xyz:URANIUM |
| Indices | xyz:SPY |
| Forex | xyz:EUR, xyz:GBP, xyz:JPY |
If a user says "buy NVDA" or "trade GOLD", use
xyz:NVDA/xyz:GOLD. These are real-world assets, not crypto.
All 13 commodity markets are HIP-3 builder-deployed perps. Their symbols use the xyz: prefix (e.g. xyz:GOLD, xyz:CL), NOT standard formats like XAU, XAG, or WTI.
Full commodity list: GOLD, SILVER, COPPER, PLATINUM, PALLADIUM, ALUMINIUM, CL (WTI crude), BRENTOIL, NATGAS, TTF (EU gas), CORN, WHEAT, URANIUM.
Key gotcha: HIP-3 assets are NOT included in allMids (the standard price feed). This means:
hl_market(coin="xyz:GOLD") may return no price or fail to find the assethl_market(dex="xyz") lists all builder markets but may not include mid pricesThe reliable way to get commodity prices is hl_candles:
# Get latest gold price (use 1h candles, lookback=24 for 24h data)
hl_candles(coin="xyz:GOLD", interval="1h", lookback=24)
# Get latest copper price
hl_candles(coin="xyz:COPPER", interval="1h", lookback=24)
# Get latest silver price
hl_candles(coin="xyz:SILVER", interval="1h", lookback=24)
The close field of the most recent candle = current price. The oldest candle's open vs latest close gives 24h change.
All existing tools work with xyz:TICKER — just pass the prefixed coin name:
hl_market(coin="xyz:NVDA") # Check NVIDIA stock perp price
hl_market(dex="xyz") # List ALL available RWA/stock perps
hl_orderbook(coin="xyz:NVDA") # Check liquidity
hl_leverage(coin="xyz:NVDA", leverage=3) # Set leverage (auto-isolated)
hl_order(coin="xyz:NVDA", side="buy", size=0.5, price=188) # Limit buy 0.5 NVDA
hl_order(coin="xyz:TSM", side="buy", size=1) # Market buy 1 TSM
hl_cancel(coin="xyz:NVDA", order_id=12345678) # Cancel order
xyz:NVDAhl_market(coin="xyz:NVDA") — Check current price, leverage limitshl_leverage(coin="xyz:NVDA", leverage=3) — Set leverage (builder perps use isolated margin)hl_order(coin="xyz:NVDA", side="buy", size=0.5, price=188) — Place limit buyhl_fills() — Check if filledhl_leverage handles this automaticallydex prefix (e.g. xyz) identifies which builder deployed the perpUser: "What's the gold price?" or "Show me commodity prices" or "Oil price?"
Name → Symbol mapping:
xyz:GOLD, SILVER→xyz:SILVER, COPPER→xyz:COPPER, PLATINUM→xyz:PLATINUM, PALLADIUM→xyz:PALLADIUM, ALUMINIUM→xyz:ALUMINIUMxyz:CL, Brent→xyz:BRENTOIL, Natural Gas→xyz:NATGAS, EU Gas→xyz:TTFxyz:CORN, Wheat→xyz:WHEATxyz:URANIUMSteps:
hl_candles(coin="xyz:GOLD", interval="1h", lookback=24) — Get 24h of hourly candlesclose(last_close - first_open) / first_open * 100Do NOT use hl_market() for commodities — HIP-3 assets are not in allMids. Always use hl_candles.
Liquidity note: CL (WTI) and BRENTOIL have the highest volume. ALUMINIUM, URANIUM, CORN, WHEAT, TTF may have zero or very low liquidity — warn the user before trading these.
User: "Buy BTC" or "Long ETH with 5x"
Agent workflow:
hl_total_balance() → Check available fundshl_leverage(coin="BTC", leverage=5) → Set leveragehl_order(...) → Place orderhl_fills() → Verify fill and report resultUser: "Buy NVIDIA" or "Short TESLA"
Agent workflow:
hl_total_balance() → Check available fundshl_leverage(coin="xyz:NVDA", leverage=10) → Set leveragehl_order(coin="xyz:NVDA", ...) → Place orderhl_fills() → Verify fill and report resultUser: "Close my BTC position"
Agent workflow:
hl_account() → Get current position sizehl_order(coin="BTC", side="sell", size=X, reduce_only=true) → Close positionhl_fills() → Report PnLFor always-on bots running inside FastAPI/worker services:
get_open_orders(address)get_user_fills(address)Important: Do not treat "order disappeared from open orders" as guaranteed fill. It can also mean cancel/reject/expired. Always confirm with get_user_fills (or get_order_status when needed).
User: "Buy 100 HYPE tokens"
Agent workflow:
hl_total_balance() → Check available USDChl_spot_order(coin="HYPE", side="buy", size=100) → Buy tokenshl_balances() → Verify purchaseDeposit:
User: "Deposit $500 USDC to Hyperliquid"
Agent: hl_deposit(amount=500) → Done
Withdraw:
User: "Withdraw $100 to my Arbitrum wallet"
Agent: hl_withdraw(amount=100) → Done (5 min, 1 USDC fee)
| Type | Parameter | Behavior |
|---|---|---|
| Limit (GTC) | order_type="limit" | Rests on book until filled or cancelled |
| Market (IoC) | omit price | Immediate-or-Cancel at mid +/- 3% slippage |
| Post-Only (ALO) | order_type="alo" | Rejected if it would cross the spread |
| Fill-or-Kill | order_type="ioc" + explicit price | Fill immediately at price or cancel |
| Stop Loss | hl_tpsl_order with tpsl="sl" | Triggers when price drops to limit losses |
| Take Profit | hl_tpsl_order with tpsl="tp" | Triggers when price rises to lock gains |
Stop loss and take profit orders are trigger orders that automatically execute when the market reaches a specified price level. Use these to manage risk and lock in profits without monitoring positions 24/7.
trigger_px, order activatesUse case: Limit losses on a position by automatically exiting if price moves against you.
Example: You're long BTC at $95,000 and want to exit if it drops below $90,000.
hl_tpsl_order(coin="BTC", side="sell", size=0.1, trigger_px=90000, tpsl="sl")
Use case: Lock in gains by automatically exiting when price reaches your profit target.
Example: You're long ETH at $3,000 and want to take profit at $3,500.
hl_tpsl_order(coin="ETH", side="sell", size=1.0, trigger_px=3500, tpsl="tp")
By default, TP/SL orders execute as market orders when triggered (instant fill, possible slippage).
For more control, use a limit order when triggered:
hl_tpsl_order(
coin="BTC",
side="sell",
size=0.1,
trigger_px=90000,
tpsl="sl",
is_market=false,
limit_px=89900
)
Trade-off: Limit orders avoid slippage but may not fill in fast-moving markets.
For short positions, reverse the side parameter:
Stop loss on short (exit if price rises):
hl_tpsl_order(coin="BTC", side="buy", size=0.1, trigger_px=98000, tpsl="sl")
Take profit on short (exit if price drops):
hl_tpsl_order(coin="BTC", side="buy", size=0.1, trigger_px=92000, tpsl="tp")
reduce_only=true (default) - ensures TP/SL only closes positions, never opens new oneshl_open_orders to verify TP/SL orders are active| Mistake | Problem | Solution |
|---|---|---|
| Wrong side | SL buys instead of sells | Long position → side="sell" for SL/TP |
| Size too large | TP/SL opens new position | Set size ≤ position size, use reduce_only=true |
| Trigger = limit | Confusion about prices | trigger_px = when to activate, limit_px = execution price |
| No SL on leverage | Liquidation risk | Always set stop loss on leveraged positions |
| Aspect | Perps | Spot |
|---|---|---|
| Tool | hl_order | hl_spot_order |
| Leverage | Yes (up to asset max) | No |
| Funding | Paid/received every hour | None |
| Short selling | Yes (native) | Must own tokens to sell |
| Check positions | hl_account | hl_balances |
| Error | Fix |
|---|---|
| "Unknown perp asset" | Check coin name. Crypto: "BTC", "ETH". Stocks: "xyz:NVDA", "xyz:TSLA" |
| "Insufficient margin" | Use hl_total_balance to check funds. Reduce size or add more USDC |
| "Order must have minimum value of $10" | Increase size. Formula: size × price ≥ $10 |
| "Size too small" | BTC min is typically 0.001. Check asset's szDecimals |
| "Order would cross" | ALO order rejected. Use regular limit order instead |
| "User or wallet does not exist" | Deposit USDC first with hl_deposit(amount=500) |
| "Minimum deposit is 5 USDC" | Hyperliquid requires at least $5 per deposit |
| "Policy violation" | Load wallet-policy skill and propose wildcard policy |
| "Action disabled when unified account is active" | Transfers blocked in unified mode (default). Just place orders directly |
| "'side' must be one of: buy/sell ..." | You passed an unrecognized direction. Use "buy" or "sell". See Side Parameter Convention above |
sickn33/antigravity-awesome-skills
moizibnyousaf/ai-agent-skills
github/awesome-copilot