uniswap-v4-lp
Uniswap V4 LP Skill
Manage concentrated liquidity positions on Uniswap V4 (Base chain).
Features
- Add liquidity to V4 pools
- Remove liquidity / collect fees
- Monitor position health
- Rebalance when out of range
- Auto-compound fees back into liquidity (set-and-forget)
Requirements
- Private key in environment (
PRIVATE_KEYorNET_PRIVATE_KEY) - Node.js 18+
- viem package
Quick Start
# Install dependencies
cd scripts && npm install
# Add liquidity ($20 test position)
node add-liquidity.mjs --amount 20 --range 25
# Check position
node check-position.mjs --token-id <ID>
# Monitor if in range
node monitor-position.mjs --token-id <ID>
# Collect fees (without removing liquidity)
node collect-fees.mjs --token-id <ID>
# Auto-compound fees → liquidity
# Two strategies: DOLLAR (default) or TIME
# Dollar strategy: compound when fees exceed $5 (default)
node auto-compound.mjs --token-id <ID>
node auto-compound.mjs --token-id <ID> --min-usd 20 # custom threshold
# Time strategy: compound on schedule (skip only if fees < gas)
node auto-compound.mjs --token-id <ID> --strategy time
# Loop mode: run continuously
node auto-compound.mjs --token-id <ID> --strategy dollar --loop --interval 3600 --min-usd 50
node auto-compound.mjs --token-id <ID> --strategy time --loop --interval 14400
# Preview
node auto-compound.mjs --token-id <ID> --dry-run
# Compound & Harvest: split fees — compound some, harvest rest as USDC
node compound-and-harvest.mjs --token-id <ID> \
--harvest-address 0xYOUR_ADDRESS --compound-pct 50
node compound-and-harvest.mjs --token-id <ID> \
--harvest-address 0xYOUR_ADDRESS --compound-pct 70 --slippage 3
node compound-and-harvest.mjs --token-id <ID> \
--harvest-address 0xYOUR_ADDRESS --dry-run
# Remove liquidity (partial)
node remove-liquidity.mjs --token-id <ID> --percent 50
# Burn position (100% removal + burn NFT)
node burn-position.mjs --token-id <ID>
# Rebalance (remove + re-add at current price)
node rebalance.mjs --token-id <ID> --range 25
Base Chain Contracts
| Contract | Address |
|---|---|
| PoolManager | 0x498581ff718922c3f8e6a244956af099b2652b2b |
| PositionManager | 0x7c5f5a4bbd8fd63184577525326123b519429bdc |
| StateView | 0xa3c0c9b65bad0b08107aa264b0f3db444b867a71 |
| Permit2 | 0x000000000022D473030F116dDEE9F6B43aC78BA3 |
Security Notes
- Always use private mempools (Flashbots/MEV Blocker) for LP operations
- Set reasonable slippage (1-3% for volatile pairs)
- Monitor positions - rebalance when 80% to range edge
- Start small - test with minimal amounts first
Pool-Specific: AXIOM/WETH
const AXIOM_WETH_POOL = {
poolId: '0x10a0b8eba9d4e0f772c8c47968ee819bb4609ef4454409157961570cdce9a735',
token0: '0x4200000000000000000000000000000000000006', // WETH
token1: '0xf3Ce5dDAAb6C133F9875a4a46C55cf0b58111B07', // AXIOM
fee: 0x800000, // ⚠️ DYNAMIC_FEE_FLAG - Clanker hook controls fee
tickSpacing: 200,
hooks: '0xb429d62f8f3bffb98cdb9569533ea23bf0ba28cc' // Clanker hook
};
⚠️ Critical: Dynamic Fee Pools
Many V4 pools (especially Clanker-deployed) use dynamic fees. The fee in PoolKey is 0x800000 (bit 23 = DYNAMIC_FEE_FLAG), NOT a percentage.
How to verify:
// Hash your PoolKey and compare to known poolId
const poolId = keccak256(abi.encode(currency0, currency1, fee, tickSpacing, hooks));
If your hash doesn't match, check if the pool uses 0x800000 as fee.
Auto-Compound Strategies
Dollar Strategy (--strategy dollar, default)
Accumulate fees in wallet, only compound when they hit a USD threshold. Best for: low-volume pools, expensive chains, or patient LPs.
# Compound when fees reach $50
node auto-compound.mjs --token-id <ID> --strategy dollar --loop --interval 3600 --min-usd 50
| Flag | Default | Description |
|---|---|---|
--min-usd |
5 | Min USD fee value to trigger compound |
--min-gas-multiple |
3 | Fees must be ≥ Nx gas cost |
--interval |
3600 | Seconds between checks in loop mode |
Time Strategy (--strategy time)
Compound on a fixed schedule. Skips only if fees < gas cost. Best for: high-volume pools where fees accumulate quickly.
# Compound every 4 hours
node auto-compound.mjs --token-id <ID> --strategy time --loop --interval 14400
| Flag | Default | Description |
|---|---|---|
--min-gas-multiple |
3 | Fees must be ≥ Nx gas cost (safety floor) |
--interval |
3600 | Seconds between compounds in loop mode |
When to Use Which
| Pool Volume | Strategy | Interval | Min USD |
|---|---|---|---|
| >$1M/day | time | 4-6h | n/a |
| $100K-$1M/day | dollar | 1h | $10-25 |
| <$100K/day | dollar | 4h | $50+ |
Both strategies always enforce a gas floor — you'll never burn money on gas.
Compound & Harvest
Split LP fees: compound a percentage back into the position and harvest the rest as USDC.
How It Works
- Collect all accrued fees (DECREASE_LIQUIDITY with 0 + CLOSE_CURRENCY)
- Split fees by compound-pct (default 50/50)
- Compound the compound portion back into the position (INCREASE_LIQUIDITY + SETTLE_PAIR)
- Swap the harvest portion of both tokens to USDC via Uniswap V3 SwapRouter02
- WETH → USDC direct (0.05% pool)
- Meme tokens → WETH → USDC multi-hop (tries 1%, 0.3%, 0.05% fee tiers)
- Transfer all USDC to the harvest address
Usage
# Preview (dry run)
node compound-and-harvest.mjs --token-id 1078751 \
--harvest-address 0xcbC7E8A39A0Ec84d6B0e8e0dd98655F348ECD44F --dry-run
# 50/50 split (default)
node compound-and-harvest.mjs --token-id 1078751 \
--harvest-address 0xcbC7E8A39A0Ec84d6B0e8e0dd98655F348ECD44F
# 70% compound, 30% harvest
node compound-and-harvest.mjs --token-id 1078751 \
--harvest-address 0xcbC7E8A39A0Ec84d6B0e8e0dd98655F348ECD44F \
--compound-pct 70
# 100% harvest (take all fees as USDC, no compounding)
node compound-and-harvest.mjs --token-id 1078751 \
--harvest-address 0xcbC7E8A39A0Ec84d6B0e8e0dd98655F348ECD44F \
--compound-pct 0
# Custom slippage
node compound-and-harvest.mjs --token-id 1078751 \
--harvest-address 0xcbC7E8A39A0Ec84d6B0e8e0dd98655F348ECD44F \
--slippage 3
Flags
| Flag | Default | Description |
|---|---|---|
--token-id |
required | LP NFT token ID |
--harvest-address |
required | Address to receive harvested USDC |
--compound-pct |
50 | Percentage to compound (0-100) |
--slippage |
1 | Slippage tolerance for swaps (%) |
--dry-run |
false | Preview without executing |
--rpc |
env/default | Base RPC URL |
Swap Routing
- Uses Uniswap V3 SwapRouter02 (
0x2626664c2603336E57B271c5C0b26F421741e481) - WETH → USDC:
exactInputSinglewith 500 (0.05%) fee tier - Other tokens → USDC:
exactInputmulti-hop through WETH, auto-tries fee tiers 10000/3000/500
🌾 Clanker Harvest — Full Treasury Pipeline
The killer feature: a complete fee management pipeline for any Clanker-launched token.
Clanker tokens have two fee sources:
- Clanker protocol fees — stored in a separate fee contract, must be claimed
- LP position fees — accrued in the V4 position, collected via DECREASE
clanker-harvest.mjs handles both in a single modular pipeline.
Quick Start
# Just claim Clanker protocol fees (no LP, no swap)
node clanker-harvest.mjs --token 0xTOKEN
# Claim + compound 100% into LP
node clanker-harvest.mjs --token 0xTOKEN --token-id 12345 --compound-pct 100
# Claim + harvest 100% as USDC to vault
node clanker-harvest.mjs --token 0xTOKEN --harvest-address 0xVAULT --compound-pct 0
# 50/50 split — compound half, harvest half
node clanker-harvest.mjs --token 0xTOKEN --token-id 12345 \
--harvest-address 0xVAULT --compound-pct 50
# 80% compound / 20% harvest, only if fees > $10
node clanker-harvest.mjs --token 0xTOKEN --token-id 12345 \
--harvest-address 0xVAULT --compound-pct 80 --min-usd 10
# Use a config file (perfect for cron)
node clanker-harvest.mjs --config harvest-config.json
Pipeline Steps
| Step | What | When |
|---|---|---|
| 1. Claim | Claim WETH + token from Clanker fee contract | Always (unless --skip-claim) |
| 2. Collect LP | Collect accrued fees from V4 position | If --token-id set (unless --skip-lp) |
| 3. Threshold | Check total USD value against --min-usd |
If threshold set |
| 4. Compound | Add X% back into LP position | If --compound-pct > 0 and --token-id set |
| 5. Swap | Swap remaining WETH to USDC | If --compound-pct < 100 and --harvest-address set |
| 6. Transfer | Send USDC to vault address | If USDC was swapped |
Flags
| Flag | Default | Description |
|---|---|---|
--token |
required | Clanker token address |
--token-id |
optional | V4 LP position NFT ID (needed for compound/LP) |
--harvest-address |
optional | Vault address for USDC (needed for harvest) |
--compound-pct |
100 | % to compound back (0 = all harvest, 100 = all compound) |
--min-usd |
0 | Min USD fee value to act (0 = always) |
--slippage |
1 | Swap slippage % |
--fee-contract |
0xf362... | Clanker fee storage contract |
--skip-claim |
false | Skip Clanker fee claim |
--skip-lp |
false | Skip LP fee collection |
--config |
optional | JSON config file path |
--dry-run |
false | Simulate without executing |
Config File
For cron jobs, use a JSON config:
{
"token": "0xYOUR_TOKEN",
"tokenId": "12345",
"harvestAddress": "0xYOUR_VAULT",
"compoundPct": 50,
"minUsd": 10,
"slippage": 1
}
Clanker Fee Contract
| Function | Description |
|---|---|
claim(feeOwner, token) |
Claim fees for a specific token |
availableFees(feeOwner, token) |
Check pending fee balance |
Contract: 0xf3622742b1e446d92e45e22923ef11c2fcd55d68
Two separate claims needed (WETH + token) — the script handles both automatically.
Standalone Claim Script
For simpler use cases, claim-clanker-fees.mjs just claims without any LP operations:
# Check available fees (dry run)
node claim-clanker-fees.mjs --token 0xTOKEN --dry-run
# Claim both WETH and token fees
node claim-clanker-fees.mjs --token 0xTOKEN
Self-Sustaining Agent Economics
The core idea: agents launched on Clanker can fund their own infrastructure from LP yield.
LP Fees + Clanker Fees
↓
┌────┴────┐
│ │
Compound Harvest
(grow LP) (→ USDC → pay for LLM, RPC, hosting)
Set up a cron job with --min-usd threshold and the agent only acts when it's profitable to do so.
LP Range Alert
Monitor your positions and get alerted when price moves near or out of range.
Quick Start
# Check single position
node lp-range-alert.mjs --positions 1078751
# Check multiple positions
node lp-range-alert.mjs --positions 1078751,1078720,1078695
# Dry run (check without saving state)
node lp-range-alert.mjs --positions 1078751 --dry-run
# JSON output (for cron/automation)
node lp-range-alert.mjs --positions 1078751 --json
# Custom alert threshold (alert at 20% from edge instead of 15%)
node lp-range-alert.mjs --positions 1078751 --near-edge-pct 20
# Force alert even if status unchanged
node lp-range-alert.mjs --positions 1078751 --force
Config File
For cron jobs, use a JSON config (lp-alert-config.example.json):
{
"positions": ["1078751", "1078720", "1078695"],
"telegramChat": "YOUR_CHAT_ID",
"nearEdgePercent": 15,
"rpcUrl": "https://mainnet.base.org"
}
Run with: node lp-range-alert.mjs --config lp-alert-config.json
Alert Statuses
| Status | Meaning | Action |
|---|---|---|
OK |
In range, healthy | None needed |
NEAR_LOWER_EDGE |
Within 15% of lower bound | Monitor, prepare to rebalance down |
NEAR_UPPER_EDGE |
Within 15% of upper bound | Monitor, prepare to rebalance up |
OUT_OF_RANGE |
Price outside position range | Rebalance immediately (not earning fees!) |
State Persistence
State is saved to ~/.lp-alert-state.json to prevent duplicate alerts.
- Only alerts on status changes (not on every check)
- Use
--forceto alert regardless of previous state
Flags
| Flag | Default | Description |
|---|---|---|
--positions |
required | Comma-separated token IDs |
--config |
optional | JSON config file path |
--telegram-chat |
optional | Chat ID for Telegram alerts |
--near-edge-pct |
15 | Alert threshold (% from edge) |
--dry-run |
false | Check without saving state |
--json |
false | JSON output |
--force |
false | Alert even if status unchanged |
--rpc |
env/default | Custom RPC URL |
Recommended Cron Setup
# Check every 30 minutes
*/30 * * * * cd /path/to/scripts && node lp-range-alert.mjs --config lp-alert-config.json >> /tmp/lp-alerts.log 2>&1
Position Strategy Recommendations
For ~$20-1000 positions:
- Range: ±25% from current price
- Rebalance: When price hits 80% of boundary
- Check: Every 4-6 hours
- Expected APR: 15-30%