Use when the user wants to swap tokens on Base Sepolia (or Base mainnet) using Uniswap V3. Handles any ERC-20 token pair including ETH wrapping/unwrapping. The agent discovers pools, gets quotes, constructs calldata, and executes swaps via keypo-wallet. Also use when the user says "swap", "trade", "exchange tokens", "buy USDC", "sell WETH", or asks about Uniswap liquidity or pricing on Base. Requires Foundry (cast) for read calls and keypo-wallet for transaction execution.
Swap any ERC-20 token pair on Uniswap V3. This skill teaches the agent to discover the right pool, quote the expected output, construct the swap calldata, and execute via keypo-wallet — for any token pair, not just predetermined ones.
| Contract | Address |
|---|---|
| V3 Factory | 0x4752ba5DBc23f44D87826276BF6Fd6b1C372aD24 |
| SwapRouter | 0x94cC0AaC535CCDB3C01d6787D6413C739ae12bc4 |
| QuoterV2 | 0xC5290058841028F1614F3A6F0F5816cAd0df5E27 |
| WETH | 0x4200000000000000000000000000000000000006 |
| Contract | Address |
|---|
| V3 Factory | 0x33128a8fC17869897dcE68Ed026d694621f6FDfD |
| SwapRouter | 0x2626664c2603336E57B271c5C0b26F421741e481 |
| QuoterV2 | 0x3d4e44Eb1374240CE5F1B871ab261CD16335B76a |
| WETH | 0x4200000000000000000000000000000000000006 |
| Token | Address | Decimals |
|---|---|---|
| USDC | 0x036CbD53842c5426634e7929541eC2318f3dCF7e | 6 |
| WETH | 0x4200000000000000000000000000000000000006 | 18 |
For tokens not listed here, ask the user for the contract address, or use the contract-learner skill to look up a token by address.
Follow these steps in order. Each step is a separate cast call or cast calldata command. Do not skip steps — the agent must gather information before constructing the transaction.
Determine the tokenIn and tokenOut contract addresses. If the user says "ETH", use the WETH address — Uniswap V3 only works with ERC-20 tokens, not native ETH.
For each token, confirm its decimals:
cast call <token-address> "decimals()(uint8)" --rpc-url https://sepolia.base.org
You need decimals to convert human-readable amounts (e.g. "0.001 ETH", "50 USDC") to raw integers:
1000000000000000 (0.001 × 10^18)50000000 (50 × 10^6)If you don't know a token's address, ask the user. Do not guess token addresses.
Uniswap V3 pools are keyed by (tokenA, tokenB, fee). The fee tiers are:
| Fee | Basis Points | Typical Use |
|---|---|---|
100 | 0.01% | Stablecoin pairs |
500 | 0.05% | Stable/major pairs |
3000 | 0.3% | Most pairs |
10000 | 1% | Exotic/volatile pairs |
Query the factory for each fee tier until you find a pool (non-zero address):
# Try 500 first (most common for major pairs)
cast call <FACTORY> "getPool(address,address,uint24)(address)" <tokenIn> <tokenOut> 500 --rpc-url https://sepolia.base.org
# If zero, try 3000
cast call <FACTORY> "getPool(address,address,uint24)(address)" <tokenIn> <tokenOut> 3000 --rpc-url https://sepolia.base.org
# If zero, try 10000
cast call <FACTORY> "getPool(address,address,uint24)(address)" <tokenIn> <tokenOut> 10000 --rpc-url https://sepolia.base.org
# If zero, try 100
cast call <FACTORY> "getPool(address,address,uint24)(address)" <tokenIn> <tokenOut> 100 --rpc-url https://sepolia.base.org
A return value of 0x0000000000000000000000000000000000000000 means no pool exists at that fee tier. If no pool exists at any fee tier, tell the user there is no Uniswap V3 liquidity for this pair on this chain.
On testnets, liquidity may be very thin or nonexistent for many pairs. Common testnet pairs with liquidity: WETH/USDC.
Once you find a pool address, verify it has liquidity:
cast call <pool-address> "liquidity()(uint128)" --rpc-url https://sepolia.base.org
If liquidity is 0, the pool exists but has no liquidity — the swap will fail. Tell the user and try the next fee tier.
Use QuoterV2 to simulate the swap and get the expected output amount. This is a read-only call — no gas needed.
# For exact input (you know how much you're putting in):
cast call <QUOTER_V2> \
"quoteExactInputSingle((address,address,uint256,uint24,uint160))(uint256,uint160,uint32,uint256)" \
"(<tokenIn>,<tokenOut>,<amountIn>,<fee>,0)" \
--rpc-url https://sepolia.base.org
The tuple parameter order is: (tokenIn, tokenOut, amountIn, fee, sqrtPriceLimitX96). Set sqrtPriceLimitX96 to 0 for no price limit.
The return values are: (amountOut, sqrtPriceX96After, initializedTicksCrossed, gasEstimate).
Convert amountOut to human-readable using the output token's decimals and show it to the user before executing. For example: "Swapping 0.001 WETH → expected ~2.34 USDC. Proceed?"
If the quote call reverts, the pool likely has insufficient liquidity for the requested amount. Try a smaller amount or a different fee tier.
Before executing, verify the wallet has enough of the input token:
# Check tokenIn balance
cast call <tokenIn> "balanceOf(address)(uint256)" <wallet-address> --rpc-url https://sepolia.base.org
# Check current allowance for the SwapRouter
cast call <tokenIn> "allowance(address,address)(uint256)" <wallet-address> <SWAP_ROUTER> --rpc-url https://sepolia.base.org
If the balance is insufficient, tell the user. If the allowance is less than amountIn, you'll need to approve in the same batch (see Step 6).
If allowance is sufficient, just swap:
SWAP_DATA=$(cast calldata \
"exactInputSingle((address,address,uint24,address,uint256,uint256,uint160))" \
"(<tokenIn>,<tokenOut>,<fee>,<recipient>,<amountIn>,<amountOutMinimum>,0)")
keypo-wallet send --key <key-name> --to <SWAP_ROUTER> --data $SWAP_DATA
If allowance is insufficient, approve + swap in one batch:
APPROVE_DATA=$(cast calldata "approve(address,uint256)" <SWAP_ROUTER> <amountIn>)
SWAP_DATA=$(cast calldata \
"exactInputSingle((address,address,uint24,address,uint256,uint256,uint160))" \
"(<tokenIn>,<tokenOut>,<fee>,<recipient>,<amountIn>,<amountOutMinimum>,0)")
echo "[
{\"to\": \"<tokenIn>\", \"value\": \"0\", \"data\": \"$APPROVE_DATA\"},
{\"to\": \"<SWAP_ROUTER>\", \"value\": \"0\", \"data\": \"$SWAP_DATA\"}
]" | keypo-wallet batch --key <key-name> --calls -
Native ETH must be wrapped to WETH first. Then approve + swap:
DEPOSIT_DATA=$(cast calldata "deposit()")
APPROVE_DATA=$(cast calldata "approve(address,uint256)" <SWAP_ROUTER> <amountIn>)
SWAP_DATA=$(cast calldata \
"exactInputSingle((address,address,uint24,address,uint256,uint256,uint160))" \
"(<WETH>,<tokenOut>,<fee>,<recipient>,<amountIn>,<amountOutMinimum>,0)")
echo "[
{\"to\": \"<WETH>\", \"value\": \"<amountIn>\", \"data\": \"$DEPOSIT_DATA\"},
{\"to\": \"<WETH>\", \"value\": \"0\", \"data\": \"$APPROVE_DATA\"},
{\"to\": \"<SWAP_ROUTER>\", \"value\": \"0\", \"data\": \"$SWAP_DATA\"}
]" | keypo-wallet batch --key <key-name> --calls -
The deposit() call wraps ETH → WETH. The value field on the deposit call must equal amountIn (in wei). All three calls execute atomically.
Swap tokenIn → WETH, then unwrap. Set the swap recipient to the wallet's own address, then unwrap:
SWAP_DATA=$(cast calldata \
"exactInputSingle((address,address,uint24,address,uint256,uint256,uint160))" \
"(<tokenIn>,<WETH>,<fee>,<wallet-address>,<amountIn>,<amountOutMinimum>,0)")
WITHDRAW_DATA=$(cast calldata "withdraw(uint256)" <expectedWethOut>)
echo "[
{\"to\": \"<tokenIn>\", \"value\": \"0\", \"data\": \"$APPROVE_DATA\"},
{\"to\": \"<SWAP_ROUTER>\", \"value\": \"0\", \"data\": \"$SWAP_DATA\"},
{\"to\": \"<WETH>\", \"value\": \"0\", \"data\": \"$WITHDRAW_DATA\"}
]" | keypo-wallet batch --key <key-name> --calls -
After the transaction succeeds, check the output token balance:
keypo-wallet balance --key <key-name> --token <tokenOut>
Or for native ETH:
keypo-wallet balance --key <key-name>
The exactInputSingle function takes a tuple with these fields in order:
| Field | Type | Description |
|---|---|---|
tokenIn | address | Input token contract address |
tokenOut | address | Output token contract address |
fee | uint24 | Pool fee tier (100, 500, 3000, or 10000) |
recipient | address | Address to receive output tokens (usually the wallet itself) |
amountIn | uint256 | Amount of input token (raw integer, adjusted for decimals) |
amountOutMinimum | uint256 | Minimum acceptable output (set to ~95-98% of quoted amount for slippage protection) |
sqrtPriceLimitX96 | uint160 | Price limit — set to 0 for no limit |
Note: The SwapRouter on Base Sepolia uses the V3 SwapRouter (not SwapRouter02). Its exactInputSingle does NOT include a deadline parameter in the tuple — the deadline is handled differently on this deployment. If you get encoding errors, check the router's ABI with cast interface.
Never set amountOutMinimum to 0 in production — this allows any output amount including near-zero (sandwich attack). Calculate a reasonable minimum:
# 3% slippage tolerance on testnet
python3 -c "
quoted = <amountOut-from-step-4>
min_out = int(quoted * 0.97)
print(min_out)
"
On testnet, 3-5% slippage is reasonable due to thin liquidity. On mainnet, 0.5-1% is typical.
If no direct pool exists for your pair but both tokens have pools with a common intermediary (usually WETH), you can route through multiple pools using exactInput:
# Example: TokenA → WETH → TokenB
# Path is encoded as: tokenA + fee1 + WETH + fee2 + tokenB (packed bytes)
# Encode the path
PATH=$(python3 -c "
tokenA = '<tokenA-address>'[2:] # remove 0x
fee1 = '<fee1-hex>' # e.g. '0001f4' for 500
weth = '<WETH-address>'[2:]
fee2 = '<fee2-hex>'
tokenB = '<tokenB-address>'[2:]
print('0x' + tokenA + fee1 + weth + fee2 + tokenB)
")
SWAP_DATA=$(cast calldata \
"exactInput((bytes,address,uint256,uint256))" \
"($PATH,<recipient>,<amountIn>,<amountOutMinimum>)")
Fee hex encoding: 100 → 000064, 500 → 0001f4, 3000 → 000bb8, 10000 → 002710.
Multi-hop is more complex. Only use it when no direct pool exists. Always try direct pools first.
"Pool not found" — No pool exists at any fee tier for this pair on this chain. On testnets, many pairs have no liquidity. Suggest the user try a different pair or check mainnet.
Quote reverts — The pool exists but doesn't have enough liquidity for the requested amount. Try a smaller amount.
Swap reverts with "STF" — "Safe Transfer From" failed. The approval didn't go through, or the wallet doesn't have enough tokens. Check allowance and balance.
Swap reverts with "TF" — "Transfer Failed" on output. Rare — usually means the output token has transfer restrictions.
Wrong decimals — Always call decimals() on both tokens. Getting this wrong sends the wrong amount (potentially 10^12x off).
"Too little received" — amountOutMinimum was set too high relative to actual pool price. This can happen if the quote is stale. Re-quote and use a slightly larger slippage tolerance.
amountOutMinimum to 0 except on testnets with trivial amounts.amountIn exactly, not type(uint256).max, to limit exposure if the router contract is compromised.