Learn-skills.dev hodlmm-bin-guardian
Monitors Bitflow HODLMM bins to keep LP positions in the active earning range. Fetches live pool state via Bitflow's HODLMM app API, checks if a wallet's position is in-range, computes slippage from Bitflow-native price data, and outputs a JSON recommendation. Read-only — rebalance actions require explicit human approval.
git clone https://github.com/NeverSight/learn-skills.dev
T=$(mktemp -d) && git clone --depth=1 https://github.com/NeverSight/learn-skills.dev "$T" && mkdir -p ~/.claude/skills && cp -r "$T/data/skills-md/aibtcdev/skills/hodlmm-bin-guardian" ~/.claude/skills/neversight-learn-skills-dev-hodlmm-bin-guardian && rm -rf "$T"
data/skills-md/aibtcdev/skills/hodlmm-bin-guardian/SKILL.mdHODLMM Bin Guardian
Monitors Bitflow HODLMM (DLMM) bins to keep LP positions in the active earning range.
What it does
Fetches live Bitflow HODLMM pool state, the user's actual LP position bins (via wallet address), and compares the user's bin range against the active bin to determine if the position is earning fees. Volume, TVL, APR, and token prices are sourced directly from Bitflow's HODLMM app API — no external oracles. Slippage is measured as the deviation between the HODLMM active-bin price and Bitflow's own reported token price. Also checks estimated gas cost and cooldown before recommending REBALANCE.
Why agents need it
HODLMM positions stop earning fees the moment the market price moves outside the deposited bin range. This skill gives an autonomous agent a reliable, safe-to-run check that surfaces out-of-range positions and flags them for human-approved rebalancing — without ever spending funds autonomously.
Safety notes
- Read-only. No transactions are submitted.
- Mainnet-only. Bitflow HODLMM API does not support testnet.
- Refuses to recommend rebalance if 24h pool volume < $10,000 USD.
- Refuses to recommend rebalance if slippage > 0.5% (HODLMM bin price vs Bitflow app price).
- Any actual rebalance (add/withdraw liquidity) requires explicit human approval before execution.
- All price data sourced from Bitflow APIs only — no external oracles.
Commands
doctor
Checks all data sources: Bitflow HODLMM API, Bitflow Bins API, Bitflow App Pools API, and Hiro Stacks API.
bun run hodlmm-bin-guardian/hodlmm-bin-guardian.ts doctor
install-packs
No additional packs required — uses Bitflow and Hiro public HTTP APIs directly.
bun run hodlmm-bin-guardian/hodlmm-bin-guardian.ts install-packs
run
Checks the LP position in the default sBTC HODLMM pool (dlmm_1) and outputs a recommendation. Pass
--wallet to enable the real in-range check against actual position bins.
# Full check with wallet (recommended) bun run hodlmm-bin-guardian/hodlmm-bin-guardian.ts run --wallet SP1234... bun run hodlmm-bin-guardian/hodlmm-bin-guardian.ts run --wallet SP1234... --pool-id dlmm_1 # Pool-only check (no position check — in_range will be null) bun run hodlmm-bin-guardian/hodlmm-bin-guardian.ts run
Live terminal output
doctor (all 4 sources reachable)
{ "status": "ok", "checks": [ { "name": "Bitflow HODLMM API", "ok": true, "detail": "8 pools found, dlmm_1 active bin: 504" }, { "name": "Bitflow Bins API (dlmm_1)", "ok": true, "detail": "active_bin_id=504, 1001 bins" }, { "name": "Bitflow App Pools API", "ok": true, "detail": "dlmm_1 TVL: $77,142.99, vol_24h: $126,045, APR: 17.72%" }, { "name": "Hiro Stacks API (fees)", "ok": true, "detail": "2 µSTX/byte" } ], "message": "All data sources reachable. Ready to run." }
run --wallet (wallet has no dlmm_1 position, slippage gate active)
{ "status": "success", "action": "HOLD — position out of range but rebalance blocked: price slippage 1.86% > 0.5% cap.", "data": { "in_range": false, "active_bin": 504, "user_bin_range": null, "can_rebalance": false, "refusal_reasons": [ "price slippage 1.86% > 0.5% cap" ], "slippage_ok": false, "slippage_pct": 1.8595, "bin_price_raw": 66459654464, "pool_price_usd": 66459.65, "market_price_usd": 65246.37, "slippage_source": "bitflow-app-price-vs-hodlmm-active-bin", "gas_ok": true, "gas_estimated_stx": 0.0144, "cooldown_ok": true, "cooldown_remaining_h": 0, "last_rebalance_at": null, "volume_ok": true, "volume_24h_usd": 126045, "liquidity_usd": 77143, "apr_24h_pct": 17.72, "pool_id": "dlmm_1", "pool_name": "sBTC-USDCx-LP", "fee_bps": 30, "position_note": "No position found for SP219TWC8G12CSX5AB093127NC82KYQWEH8ADD1AY in pool dlmm_1." }, "error": null }
run --wallet (active position in range, all gates pass)
{ "status": "success", "action": "HOLD — position in range at active bin 504. APR (24h): 17.72%.", "data": { "in_range": true, "active_bin": 504, "user_bin_range": { "min": 500, "max": 508, "count": 3, "bins": [500, 504, 508] }, "can_rebalance": true, "refusal_reasons": null, "slippage_ok": true, "slippage_pct": 0.04, "bin_price_raw": 66459654464, "pool_price_usd": 66459.65, "market_price_usd": 66433.0, "slippage_source": "bitflow-app-price-vs-hodlmm-active-bin", "gas_ok": true, "gas_estimated_stx": 0.0144, "cooldown_ok": true, "cooldown_remaining_h": 0, "last_rebalance_at": null, "volume_ok": true, "volume_24h_usd": 126045, "liquidity_usd": 77143, "apr_24h_pct": 17.72, "pool_id": "dlmm_1", "pool_name": "sBTC-USDCx-LP", "fee_bps": 30 }, "error": null }
Output contract
All outputs are strict JSON to stdout.
| Field | Type | Description |
|---|---|---|
| | Overall result |
| | , , or with reason |
| | if no wallet provided |
| | Pool's current active bin ID |
| | User's liquidity bin range |
| | Whether all safety gates pass |
| | Why REBALANCE is blocked |
| | Whether price deviation is within cap |
| | |
| | Raw active bin price from Bitflow bins API |
| | HODLMM derived USD price: |
| | Bitflow app reported token price in USD |
| | Price source identifier |
| | Whether estimated gas is within limit |
| | Estimated STX for 2-txn rebalance |
| | Whether cooldown has elapsed |
| | Hours until next rebalance allowed |
| | ISO timestamp of last recorded rebalance |
| | Whether 24h volume meets minimum |
| | 24h pool volume in USD |
| | Pool TVL in USD |
| | 24h fee APR from Bitflow app API |
| | Pool identifier |
| | Human-readable pool name |
| | Pool fee in basis points |
| | Present when position state needs explanation |
Data sources
| Source | Data | Endpoint |
|---|---|---|
| Bitflow HODLMM API | Pool list, active bin | |
| Bitflow Bins API | Per-bin prices (raw, for slippage) | |
| Bitflow App Pools API | TVL, 24h volume, APR, token prices, decimals | |
| Bitflow Position API | User's position bins | |
| Hiro Stacks API | STX fee estimate | |
v2 changelog (fixes from day-1 review)
In-range check — was fake, now real
Before:
inRange = isFinite(pool.active_bin) && pool.active_bin > 0 — always true.
After: Real HTTP call to
GET /api/app/v1/users/{address}/positions/{poolId}/bins. Filters bins where user_liquidity > 0, checks if active_bin_id is in that set.
Slippage — was hardcoded, now live and fully Bitflow-native
Before: Constant value, always passed.
After:
(bin_price_raw / 1e8) × 10^(xDec − yDec) vs Bitflow app API token price. No external oracles — all data from Bitflow endpoints.
Gas estimate — was a made-up constant, now live
Before:
gas_estimated_stx: 0.006 — hardcoded.
After:
Hiro /v2/fees/transfer × 500 bytes × 2 txns × 3× contract multiplier × 1.2 safety buffer.
Cooldown — was not tracked, now persistent
Before: No state file, cooldown always passing.
After: Reads/writes
~/.hodlmm-guardian-state.json. Returns cooldown_remaining_h on each run.
Frontmatter — stale dependency removed
Before:
requires: [bitflow] — referenced a non-existent dependency.
After:
requires: "" — fully self-contained, all data from public HTTP APIs.
Origin
Winner of AIBTC x Bitflow Skills Pay the Bills competition. Original author: @cliqueengagements Competition PR: https://github.com/BitflowFinance/bff-skills/pull/39