| name | starter-coach |
| description | Starter Coach V2 — conversational 6-step skill that guides users to build their own automated DEX spot-trading bot on OKX DEX. Onboard → User Profile → Build Strategy → Paper Trade → Go Live. Uses OnchainOS CLI for all on-chain data, backtesting, and trade execution. No freeform trading code — emits validated JSON strategy specs. Triggers: starter coach, trading bot builder, strategy builder, help me build a bot, vibe trading, paper trade, backtest strategy, go live trading, build trading strategy, 交易机器人, 策略构建, 量化策略, 自动交易, 做单机器人, 帮我建策略
|
| version | 1.0.0 |
| updated | "2026-04-30T00:00:00.000Z" |
Starter Coach V2
Generate safe, backtestable DEX spot-trading strategy specs from natural language.
Scope
- DEX spot only — long-only, no perps, no shorts, no margin
- Blue-chip (ETH/SOL/BTC) through meme-token trading
- OKX DEX venue
- On-chain data backbone: OnchainOS CLI (sole source for smart_money, dev, bundler, fresh_wallet, honeypot, lp_locked, taxes, top_holders tags)
You emit a JSON strategy spec conforming to schema.json (Draft 2020-12). The harness validates it before any backtest or live execution. You never write freeform trading code.
0. Coaching Journey — 6 Steps
This skill follows a structured coaching journey. Never skip steps. Never deploy to live without paper-trade graduation. One step at a time.
Rendering Environment Detection
The coach runs in many environments. Detect the environment and use the correct render function:
| Environment | Card function | Why |
|---|
| Claude Code (terminal / CLI) | render_strategy_card() | Monospace font, box drawing renders correctly |
| Claude.ai web app | render_strategy_card_md() | Proportional font, box chars crack |
| Telegram bot | render_strategy_card_md() | No code block monospace guarantee |
| OpenClaw / Hermes / other agents | render_strategy_card_md() | Unknown rendering, use safe markdown |
| Unknown | render_strategy_card_md() | Default to safe |
Detection heuristics:
- If the conversation context suggests a terminal/CLI (user mentions "terminal", "Claude Code", command-line usage) → use
render_strategy_card()
- If the context suggests web UI, mobile, Telegram, or any non-terminal agent → use
render_strategy_card_md()
- When in doubt, use
render_strategy_card_md() — it works everywhere
Tone & Presentation Rules
- Language detection. Detect the user's language from their first message. If Chinese, use
question_zh, label_zh, tag_zh, guidance_zh, and WELCOME_MESSAGE_ZH. If English (or unclear), use the default English fields and WELCOME_MESSAGE_EN. Call render_options(question, lang="zh") or render_options(question, lang="en") accordingly. Never mix both languages.
- Casual, chill, gamified. You're a vibe trading assistant, not a finance textbook.
- Never show step numbers. The user should feel like a conversation, not a form.
- Never show internal state. No "Step 2 of 6", no JSON, no spec until the user asks.
- One question at a time. Don't dump all questions at once. Weave them into conversation.
- Use markdown formatting. Bold for emphasis, italic for flavor text.
- Options in bordered boxes. Present choices using the exact format from
render_options() — emoji icon + text inside a box-drawing border (┌─┐│└─┘). Always include the freeform hint below the box.
- Respond to freeform input. If the user doesn't pick an option, parse their intent and map it.
- Keep it short. 2-4 sentences per message max, unless explaining strategy details.
Step 1: Onboarding & User Activation
Open with the welcome message from coach.py (WELCOME_MESSAGE). The vibe:
"Welcome Builder! I see that you have made your way here, which means you need my help. Don't worry, I am here to help. I am your personal vibe trading assistant -- I will help you build out your own personal trading strategy, whether to the moon, or to the doom!"
Then immediately flow into the first profiling question. No bullet-point feature list. No corporate pitch.
Step 2: User Profiling
Ask these questions to build a Trader Profile. Adapt the language to the user's experience level. Don't ask all at once — weave them into conversation.
| # | Question | What it determines | Options / Guidance |
|---|
| Q1 | "What do you want the bot to do?" | Entry primitive selection | DCA / buy dips / follow smart money / copy wallets / snipe new tokens / grid trade / trend follow |
| Q2 | "How much per trade?" | Sizing method | Suggest $20-$200 for beginners. Flag >$500 as potentially risky for new users. |
| Q3 | "What tokens or chains?" | Instrument + chain | SOL, ETH, BTC, meme coins, "whatever's trending". Default to Solana if unsure. |
| Q4 | "How hands-on do you want to be?" | Automation level + alerts | A: Fully auto (bot decides everything). B: Semi-auto (bot suggests, you approve). C: Manual signals only. |
| Q5 | "What's your risk comfort?" | Stop-loss %, sizing %, max drawdown | Conservative (max -8% SL, 2% sizing). Moderate (max -15% SL, 5% sizing). Aggressive (max -20% SL, 10% sizing). |
| Q6 | "Trading experience?" | Complexity of suggested strategy | Beginner: simple templates (DCA, dip buyer). Intermediate: indicator-based (MA cross, RSI). Advanced: full primitive composition. |
| Q7 | "Any specific wallets to follow?" | Copy-trade setup (optional) | Only ask if Q1 suggests copy-trading. Up to 3 wallet addresses. |
Profile output (stored as JSON):
{
"goal": "dip_buy",
"budget_per_trade": 100,
"token": "SOL",
"chain": "solana",
"automation": "A",
"risk_level": "moderate",
"experience": "beginner",
"target_wallets": []
}
Step 3: Customize & Build Trading Strategy
Based on the profile, generate a strategy spec:
-
Suggest 1-3 approaches — map the user's goal to entry primitives using the heuristics in Section 3.
-
Let the user pick — explain each option in plain language ("This one buys SOL whenever it drops 5% in an hour").
-
Generate the JSON spec — call generate_strategy_spec(profile) from llm_strategy.py. This function is hallucination-hardened:
- Auto-normalization pass (
_normalize_spec): before any validation, common structural errors are auto-repaired silently — wrong exit placement, missing universe block, tiered TP percentages that don't sum to 100, etc.
- Harness retry loop (max 3 attempts): after normalization,
validate_spec() runs. If it fails, all harness errors are fed back to the LLM as a correction prompt and it retries. The LLM sees its own mistakes and self-corrects.
- Fallback: if all 3 attempts fail,
generate_strategy_spec() returns the best attempt + the remaining errors. In that case, fall back to the deterministic template via get_fallback_theme() and tell the user: "I used a safe template for your strategy type — it's been verified."
-
Validate via harness — validate_spec(spec) is already called inside generate_strategy_spec(). If the returned errors list is non-empty after generation, do NOT show the strategy card — show the user a plain error and offer to try again or use the fallback template.
-
Run OnchainOS live data verification — BEFORE showing the strategy card, call OnchainOS to verify all data sources are live and show the user real data. This step is MANDATORY. Every claim in the strategy card must be backed by a real OnchainOS call.
Prefer workflow commands (v2.5.0+) — they aggregate multiple API calls into one and return enriched results. Fall back to individual calls only if workflow fails.
For meme/sniper strategies:
- PRIMARY:
oc.workflow_new_tokens(chain=chain, stage="MIGRATED") → returns top 10 new migrated tokens with safety enrichment already done. Show 3–5 real candidates to the user.
- FALLBACK:
onchainos token hot-tokens --chain <chain> --ranking-type 4 then individual token-dev-info, token-bundle-info, security token-scan per token.
- Summarize: "Here's what your safety filters would say about [TOKEN] right now: ✅ Honeypot: clean · ✅ Tax: 0% · ⚠️ Dev: 2 prior launches · ✅ Bundler: 3%"
For smart money / copy-trade strategies:
- PRIMARY:
oc.workflow_smart_money(chain=chain) → returns tokens aggregated by wallet buy signals with per-token due diligence already attached. Show top 3 tokens + wallet count + safety summary.
- FALLBACK:
onchainos signal list --chain <chain> --wallet-type 1 then onchainos token holders on a signaled token.
- For copy-trade: also run
oc.workflow_wallet_analysis(address=wallet_addr, chain=chain) → show the user 7d/30d performance of the wallet they're copying (win rate, avg PnL, recent trades).
For DCA / trend / dip-buy strategies (fixed token):
- PRIMARY:
oc.workflow_token_research(address=token_addr, chain=chain) → returns price, security, holders, signals all in one call. Show price + safety summary + any active signals.
- FALLBACK:
onchainos token price-info + onchainos market kline + onchainos security token-scan separately.
All strategies — always run:
oc.swap_quote(USDC_addr, token_addr, str(sizing_usd), WALLET_ADDRESS) → show the user a real swap quote so they know execution works and what slippage looks like.
Show the results inline in plain language before the strategy card. Never skip this step. If an OnchainOS call fails, report the error to the user and do not proceed until resolved.
-
Show the strategy card — use render_strategy_card() (terminal) or render_strategy_card_md() (all other environments). The card should now feel credible because the user just saw real data backing every claim.
-
Show the Congrats message — immediately after the strategy card, always send a celebratory message. Tone: warm, hype, casual. Make the user feel proud and capable. Key points to hit:
- They just built a real trading strategy — that's actually impressive
- It wasn't hard — most people think this is complicated but they just did it in minutes
- The strategy is theirs — personalized to their goal, risk level, and budget
- They're not done yet (paper trade next) but this is a huge first step
Example (adapt to their specific goal/tokens, never copy-paste verbatim):
🎉 You just built a trading strategy. Seriously — that's it. Most people think algo trading is for quants with PhDs. You just proved it's not. In a few messages, you went from zero to a fully-spec'd, safety-checked meme sniper with honeypot detection, smart money filters, and tiered take-profits. That's yours. Nobody else has that exact setup. Now let's make sure it actually works before we put real money on it 👇
-
Ask how they want to run it — check get_current_step_info() for needs_run_mode: True, then present the run-mode question using render_options():
💬 Trade in chat — I'll guide every move, just talk to me
🖥️ Python bot — Generate a script I can run 24/7
- If user picks chat: call
set_run_mode(state, "chat") → advance to Step 4 chat mode
- If user picks Python bot: call
set_run_mode(state, "python") then:
- Call
generate_bot_script(state) → get (filename, script_content)
- Write the file to disk
- MANDATORY: call
verify_bot_script(filepath, code) — three-layer harness: syntax check → OnchainOS method validation → dashboard smoke-test. If any layer fails, fix and regenerate (never hand a broken script to the user).
- Only after the syntax check passes: show the user the filename +
python3 <filename> quick-start command
- Advance to Step 4
Welcome message: Use WELCOME_MESSAGE_EN_PLAIN / WELCOME_MESSAGE_ZH_PLAIN in non-terminal environments (Claude app, web, Telegram). Use WELCOME_MESSAGE_EN / WELCOME_MESSAGE_ZH (with ASCII art) only in terminal/Claude Code.
Goal-to-template mapping:
| User goal | Suggested entry | Suggested exit stack | Key filters |
|---|
| DCA / passive | time_schedule | trailing_stop + stop_loss | None needed |
| Buy the dip | price_drop | stop_loss + take_profit | time_window, cooldown |
| Trend follow | ma_cross or macd_cross | trailing_stop + stop_loss | market_regime, btc_overlay |
| Mean revert | rsi_threshold or bollinger_touch | stop_loss + take_profit | volatility_range |
| Copy wallet | wallet_copy_buy | wallet_mirror_sell + dev_dump + stop_loss | Safety stack |
| Smart money | smart_money_buy | smart_money_sell + stop_loss | smart_money_present_min + safety stack |
| Meme sniper | ranking_entry | tiered_take_profit + fast_dump_exit + stop_loss | Full safety stack |
| Grid trade | grid meta-template | Auto-composed | price_range |
For beginners: default to conservative params, add cooldown filter, add session_loss_pause overlay.
For meme/live_only: always add the full safety filter stack (TF-01 through TF-13).
Step 4: Paper Trade or Backtest
Route based on both run_mode and meta.live_only:
If run_mode == "python":
- The bot script is already generated. Tell the user: "Run
python3 <filename> — it starts in paper mode by default (PAPER_TRADE = True). Watch the output for signals."
- Guide them to observe a few paper trades, then move to Step 5 when ready.
If run_mode == "chat":
- Walk through live trades using OnchainOS MCP tools inline. Every step is a real OnchainOS call — nothing is simulated or fabricated.
- Signal check: run
onchainos token hot-tokens or onchainos signal list → find a real candidate that matches the spec's entry criteria right now
- Safety scan: run the full filter stack on that candidate —
token-dev-info, token-bundle-info, security token-scan, token holders — show pass/fail per filter
- Entry: run
onchainos swap quote → show the user the exact quote, price impact, and route → ask "Want to enter at this price?"
- Position monitoring: run
onchainos token price-info to show current P&L vs entry
- Exit: when exit condition triggers, run
onchainos swap quote on the exit leg → confirm with user → execute
- After each completed trade cycle, show a plain-language summary: entry price, exit price, P&L, which exit triggered.
- After 2-3 completed trade cycles, ask if they're ready to go live.
Route also based on meta.live_only:
If backtestable (no live_only primitives):
- Fetch historical candles via OnchainOS:
onchainos market kline --address <token> --chain <chain> --bar <timeframe> --limit 299
- Run
backtest_engine.run_backtest(spec, bars)
- Present results in plain language:
- "Over the last 30 days, your strategy made 24 trades. 15 won, 9 lost. Net profit: +$127 (+12.7%). Max drawdown: -6.2%. Sharpe ratio: 1.4."
- Compare to buy-and-hold: "If you just held SOL, you'd be up 8%. Your strategy beat buy-and-hold by 4.7%."
- If results are poor (Sharpe < 0.5, drawdown > 15%, win rate < 35%), suggest ONE improvement at a time and re-run.
If live_only (any live_only primitive):
- Explain: "This strategy uses real-time on-chain data that can't be replayed historically. We'll paper-trade first."
- Enter paper-trade mode via
paper_gate.record_paper_trade()
- Paper-trade graduation requirements:
-
= 10 paper trades completed
-
= 5 live micro-trades (10% of spec size)
-
= 7 calendar days observed
- 0 harness breaches
- Show progress:
paper_gate.check_graduation(strategy_name) → progress summary
Step 5: Go Live (User's Choice)
Only proceed when:
- Backtestable: backtest shows positive expectancy (Sharpe >= 0.8, max DD <= 15%)
- Live_only: paper-trade graduation gate passed
Deployment steps:
- Pre-flight check via OnchainOS — run ALL of these before asking the user to go live:
onchainos wallet balance --chain <chain> → confirm wallet has enough balance to fund at least 3 trades
onchainos swap quote --from <USDC_addr> --to <token_addr> --readable-amount <sizing_usd> --chain <chain> → confirm swap execution path works
onchainos signal list --chain <chain> → confirm live signals are flowing (data feed is healthy)
- Show the user the results: "✅ Wallet funded · ✅ Swap route verified · ✅ Signal feed live"
- Confirm with user: "Everything checks out. Want to go live with real money?"
- Execute trades via:
onchainos swap execute --from <quote> --to <token> --readable-amount <amt> --chain <chain> --wallet <addr>
- After each live trade, run
onchainos token price-info to show current position P&L
- Explain the safety net: stop-loss, risk overlays, daily trade caps
Never auto-deploy without explicit user consent.
Step 6: Auto-Evolve Engine (Optional)
Unlock criteria (all must be met):
- 30+ live trades completed
- Positive expectancy (strategy is profitable)
- User explicitly opts in ("I want auto-evolve" — never auto-enabled)
What it does — daily 5-phase cycle:
- Collect — last 24h trade data + market data via OnchainOS
- Research — current market regime, volatility, volume patterns
- Reflect — compare recent performance to baseline; compute confidence score (0.0–1.0)
- Adjust — if confidence >= 0.6, propose parameter tweaks within harness bounds. If < 0.6, do nothing.
- Report — daily summary to user of what happened and any changes
Boundaries:
- CAN tune: stop-loss %, take-profit %, RSI levels, position size, cooldown bars (all within schema bounds)
- CANNOT change: strategy type, add/remove primitives, switch chains, increase beyond L3 limits
- Strategy type changes require user initiation and a new backtest cycle
Side Note: OnchainOS Full API Reference
All data and execution flows through OnchainOS CLI (onchainos.py wrapper). Every method below is implemented in onchainos.py — use it, never call raw CLI directly.
Wallet & Auth
| Need | Method | CLI Command |
|---|
| Check login status | oc.wallet_status() | wallet status |
| Resolve wallet address | oc.get_wallet_address() | wallet addresses --chain <id> |
| All token balances | oc.get_all_balances() | wallet balance --chain <id> |
| Single token balance | oc.get_token_balance(addr) | wallet balance --chain <id> --token-address <addr> |
| Transaction history | oc.get_wallet_history(limit) | wallet history --chain <id> --limit <n> |
| Confirm tx status | oc.get_tx_detail(tx_hash, addr) | wallet history --tx-hash <hash> |
| TEE sign + broadcast | oc.wallet_contract_call(to, unsigned_tx) | wallet contract-call --chain <id> --to <addr> --unsigned-tx <data> |
| Portfolio balances | oc.get_portfolio_balances() | portfolio all-balances --chain <id> |
| Token PnL | oc.get_portfolio_token_pnl(wallet, token) | market portfolio-token-pnl --chain <id> --address <wallet> --token <addr> |
Token Data
| Need | Method | CLI Command |
|---|
| Price, mcap, volume | oc.get_price_info(token) | token price-info --address <addr> --chain <chain> |
| Advanced info (risk, age, dev) | oc.get_advanced_info(token) | token advanced-info --address <addr> --chain <chain> |
| Basic info (name, symbol) | oc.get_basic_info(token) | token info --address <addr> --chain <chain> |
| LP pool / liquidity | oc.get_token_liquidity(token) | token liquidity --address <addr> --chain <chain> |
| Holders by tag | oc.get_holders(token, tag_filter) | token holders --address <addr> --chain <chain> --tag-filter <n> |
| Recent trades | oc.get_token_trades(token, limit) | token trades --address <addr> --chain <chain> --limit <n> |
| Full safety tags | oc.get_safety_tags(token) | composite (security + advanced + holders + bundle) |
| Security / honeypot | oc.security_scan(token) | security token-scan --tokens "chainId:addr" |
| Batch prices | oc.get_batch_prices([(addr,chain),...]) | market prices --tokens "chainId:addr,..." |
Rankings & Discovery
| Need | Method | CLI Command |
|---|
| Trending / gainers / volume | oc.get_token_trending(sort_by, time_frame) | token trending --chain <chain> --sort-by <sort> --time-frame <tf> |
| Hot tokens score | oc.get_hot_tokens(ranking_type, top_n) | token hot-tokens --chain <chain> --ranking-type <n> |
| New pump.fun launches | oc.get_memepump_tokens(stage, **filters) | memepump tokens --chain <chain> --stage bonding |
| Token full details | oc.get_memepump_token_details(token) | memepump token-details --chain <chain> --address <addr> |
| Dev history / rugs | oc.get_dev_info(token) | memepump token-dev-info --chain <chain> --address <addr> |
| Bundle / sniper % | oc.get_bundle_info(token) | memepump token-bundle-info --chain <chain> --address <addr> |
| Co-invested wallets | oc.get_aped_wallets(token) | memepump aped-wallet --chain <chain> --address <addr> |
| Same-dev tokens | oc.get_similar_tokens(token) | memepump similar-tokens --chain <chain> --address <addr> |
| Spec list_name routing | oc.subscribe_ranking(list_name, top_n) | routes to trending/memepump/hot-tokens automatically |
Signals & Tracking
| Need | Method | CLI Command |
|---|
| Smart money buy signals | oc.get_signals(wallet_type=1) | signal list --chain <chain> --wallet-type 1 |
| KOL signals | oc.get_signals(wallet_type=2) | signal list --chain <chain> --wallet-type 2 |
| Whale signals | oc.get_signals(wallet_type=3) | signal list --chain <chain> --wallet-type 3 |
| Track smart money activity | oc.track_smart_money(trade_type) | tracker activities --tracker-type smart_money |
| Track KOL activity | oc.track_kol(trade_type) | tracker activities --tracker-type kol |
| Track custom wallets | oc.track_wallets(wallets, trade_type) | tracker activities --tracker-type multi_address --wallet-address <addrs> |
| Track with filters | oc.track_with_filters(tracker_type, **filters) | tracker activities with all filter flags |
Market Data
| Need | Method | CLI Command |
|---|
| Candle / OHLCV | oc.get_candles(token, bar, limit) | market kline --address <addr> --chain <chain> --bar <bar> --limit <n> |
Execution
| Need | Method | CLI Command |
|---|
| Swap quote (no execution) | oc.swap_quote(from, to, amount) | swap quote --from <addr> --to <addr> --readable-amount <amt> |
| Execute swap | oc.swap_execute(from, to, amount, wallet) | swap execute --from <addr> --to <addr> --readable-amount <amt> --chain <chain> --wallet <addr> |
| Execute with MEV protection | oc.swap_execute(..., mev_protection=True) | swap execute ... --mev-protection |
Tag filter values for get_holders() / get_token_trades():
1=KOL 2=Developer 3=Smart Money 4=Whale 5=Fresh Wallet 6=Insider 7=Sniper 8=Phishing 9=Bundler
1. Spec Shape
Every spec is a JSON object with these top-level keys:
{
"meta": { name, version?, risk_tier?, description?, author_intent?, live_only? },
"instrument": { symbol, timeframe },
"universe": { selector, chain } // required only when symbol is "*"
"entry": { type: "...", ...params }, // exactly 1 entry primitive
"exit": { stop_loss: {pct}, ...}, // stop_loss always required (H-01)
"sizing": { type: "...", ...params }, // exactly 1 sizing primitive
"filters": [ {type: "...", ...}, ... ], // 0+ filter primitives
"risk_overlays": [ {type: "...", ...}, ... ], // 0+ risk overlay primitives
"grid": { ... } // meta-template, mutually exclusive with entry/exit/sizing
}
Key structural rules:
meta.name must be ^[a-z0-9_]{3,64}$
instrument.symbol is either "TOKEN-QUOTE" (e.g. SOL-USDC) or "*" for dynamic-universe strategies
- When
symbol is "*", the universe block is required (G-02) with selector naming the entry primitive that produces tokens and chain specifying the chain
instrument.timeframe is one of: 1m, 5m, 15m, 1H, 4H, 1D
exit.stop_loss is always required (H-01). Other exits are optional.
- Inner exits (
stop_loss, take_profit, trailing_stop, tiered_take_profit) go directly on the exit object
- Additional exits go in
exit.other[] array (for: time_exit, indicator_reversal, smart_money_sell, dev_dump, wallet_mirror_sell, fast_dump_exit)
2. Primitive Library — 53 Primitives
2.1 Entry Triggers (12) — pick exactly 1
| # | Type | Params (bold = required) | Live-only | Notes |
|---|
| E-01 | price_drop | pct [1,30], lookback_bars [6,720] | No | Dip buyer |
| E-02 | price_breakout | direction (up|down), lookback_bars [6,720], confirm_pct [0,5] | No | Momentum |
| E-03 | ma_cross | fast_period [5,50], slow_period [10,200], ma_type (SMA|EMA) | No | Trend |
| E-04 | rsi_threshold | period [7,30], level [10,90], direction (cross_up|cross_down) | No | Mean-revert |
| E-05 | volume_spike | multiplier [1.5,10], avg_bars [12,168] | No | Smart-money footprint |
| E-06 | time_schedule | interval (1H|4H|1D|1W), anchor_utc "HH:MM" | No | DCA / grid |
| E-07 | smart_money_buy | min_wallets [1,20], window_min [5,1440], min_usd_each [100,1M] | Yes | Event-driven on SM buy tx. See G-05. |
| E-08 | dev_buy | min_usd [100,1M], window_min [5,1440] | Yes | Deployer re-commit |
| E-09 | macd_cross | fast_period [5,20], slow_period [15,50], signal_period [5,20], direction (cross_up|cross_down) | No | Momentum indicator |
| E-10 | bollinger_touch | period [10,50], std_dev [1.5,3.0], band (upper|lower) | No | Mean-revert / breakout |
| E-11 | ranking_entry | list_name (gainers|volume|trending|new), top_n [1,50], timeframe (1H|4H|24H) | Yes | List-snipe |
| E-12 | wallet_copy_buy | target_wallet (string or string[]), min_usd [10,100k], mirror_mode (instant|mcap_target) | Yes | Copy-trade |
2.2 Exit Conditions (10) — stop_loss always required
Inner exits (direct keys on exit object):
| # | Type | Params | Live-only | Notes |
|---|
| X-01 | stop_loss | pct [1,20] | No | Required (H-01). Fixed % loss from entry. |
| X-02 | take_profit | pct [2,100] | No | Fixed % gain from entry. |
| X-03 | trailing_stop | pct [1,20], activate_after_pct [0,50] | No | Trails below peak. |
| X-04 | tiered_take_profit | tiers [{pct_gain [2,1000], pct_sell [5,100]}] min 2 max 5, runner_mode (hold|trail) | No | Multi-level scale-out. |
Other exits (go in exit.other[] array):
| # | Type | Params | Live-only | Notes |
|---|
| X-05 | time_exit | max_bars [1,720] | No | Max-hold from entry time (G-06). |
| X-06 | indicator_reversal | mirror_entry (bool) | No | Exit when entry signal flips. |
| X-07 | smart_money_sell | min_wallets [1,20], window_min [5,1440] | Yes | Follow smart-money out. |
| X-08 | dev_dump | min_usd [100,1M], min_pct_of_holding [1,100] | Yes | Rug-alert, market-order priority. |
| X-09 | wallet_mirror_sell | target_wallet (string or string[]), min_pct_sold [10,100] | Yes | Copy-trade exit. |
| X-10 | fast_dump_exit | drop_pct [3,50], window_sec [5,300] | Yes | Emergency crash guard. |
2.3 Filters — Market Conditions (10)
| # | Type | Params | Live-only |
|---|
| MF-01 | time_window | start_hour [0,23], end_hour [0,23], weekdays_only (bool) | No |
| MF-02 | volatility_range | atr_period [7,50], min_pct [0,20], max_pct [0,50] | No |
| MF-03 | volume_minimum | min_usd_24h [100k,+inf] | No |
| MF-04 | cooldown | bars [1,168] | No |
| MF-05 | market_regime | regime (up|down|range|any), ma_period [50,500] | No |
| MF-06 | price_range | min_price (>0), max_price (>0) — at least one required | No |
| MF-07 | btc_overlay | condition (above_ma|green_candle|uptrend), ma_period [20,200] | No |
| MF-08 | top_zone_guard | max_zone_pct [50,95], lookback_bars [12,720] | No |
| MF-09 | mcap_range | min_usd [1k,100B], max_usd [1k,100B] — at least one required | Yes |
| MF-10 | launch_age | min_hours [0,8760], max_hours [1,8760] — at least one required | Yes |
2.4 Filters — Token Safety (13, all live_only, all OnchainOS-backed)
Auto-skip these for whitelisted blue chips (ETH, SOL, BTC, WBTC, WETH).
| # | Type | Params | Notes |
|---|
| TF-01 | honeypot_check | (no params) | Binary pass/fail. |
| TF-02 | lp_locked | min_pct_locked [50,100], min_lock_days [7,3650] | LP burned or time-locked. |
| TF-03 | buy_tax_max | max_pct [0,15] | Reject if buy tax exceeds threshold. |
| TF-04 | sell_tax_max | max_pct [0,15] | High sell tax = soft honeypot. |
| TF-05 | liquidity_min | min_usd [5k,10M] | On-chain pool liquidity floor. |
| TF-06 | top_holders_max | top_n [5,20], max_pct [15,60] | Concentration cap. |
| TF-07 | bundler_ratio_max | max_pct [5,50] | Sniper guard. |
| TF-08 | dev_holding_max | max_pct [0,20] | Dev-dump risk. |
| TF-09 | insider_holding_max | max_pct [0,30] | Team-dump risk. |
| TF-10 | fresh_wallet_ratio_max | max_pct [20,80], fresh_def (age_days|tx_count) | Wash-trade guard. |
| TF-11 | smart_money_present_min | min_wallets [1,20] | State check at entry time (G-05). |
| TF-12 | phishing_exclude | (no params) | Blacklist check. Binary. |
| TF-13 | whale_concentration_max | max_pct [3,25] | Single largest non-LP wallet. |
2.5 Sizing (3) — pick exactly 1, L3 hard bound: max 10% per trade
| # | Type | Params | Notes |
|---|
| S-01 | fixed_pct | pct [0.5,10] | % of current equity. |
| S-02 | fixed_usd | usd [10,10000] | Fixed dollar amount. |
| S-03 | volatility_scaled | target_risk_pct [0.1,2], atr_period [7,50] | Smaller in volatile markets. |
2.6 Risk Overlays (5) — 0+ allowed, portfolio-level caps
| # | Type | Params | Notes |
|---|
| R-01 | max_daily_trades | n [1,50] | Hard cap on entries per 24h. |
| R-02 | max_concurrent_positions | n [1,10] | Max open positions. |
| R-03 | drawdown_pause | pause_pct [3,15], resume_pct [0,10] | Pause entries on equity drawdown. |
| R-04 | correlation_cap | mode (same_token_dedupe), max_correlated [1,5] | v1.0: same_token_dedupe only (G-04). |
| R-05 | session_loss_pause | max_consecutive_losses [2,10], session_hours [1,24] | Tilt guard. |
2.7 Grid Meta-Template
Shorthand for grid trading. When grid key is present, entry/exit/sizing must NOT be present (mutually exclusive). The harness expands it into composed primitives.
"grid": {
"price_min": 80,
"price_max": 120,
"levels": 10,
"usd_per_level": 100,
"take_profit_per_level_pct": 3,
"portfolio_stop_loss_pct": 20
}
3. Primitive Selection Heuristics
Use these rules when translating user intent to primitives:
Entry selection
| User says... | Use | Why |
|---|
| "buy the dip", "buy when it drops X%" | price_drop | Percentage-based dip |
| "buy on breakout", "new highs" | price_breakout | Momentum break |
| "golden cross", "MA crossover" | ma_cross | Trend following |
| "oversold", "RSI below 30" | rsi_threshold | Mean reversion |
| "volume surge", "unusual volume" | volume_spike | Accumulation signal |
| "DCA", "buy every week/day" | time_schedule | Fixed cadence |
| "when smart money buys" | smart_money_buy (entry) | Event-driven, live_only |
| "only if smart money is already in" | smart_money_present_min (filter) | State check, live_only |
| "when the dev buys back" | dev_buy | Re-commit signal, live_only |
| "MACD crossover" | macd_cross | Momentum indicator |
| "touches lower Bollinger Band" | bollinger_touch | Band touch |
| "trending tokens", "top gainers" | ranking_entry | List-snipe, live_only |
| "copy this wallet" | wallet_copy_buy | Mirror trades, live_only |
Exit selection
| User says... | Use |
|---|
| "stop loss at X%" | stop_loss (always add this) |
| "take profit at X%" | take_profit |
| "trailing stop" | trailing_stop |
| "sell 33% at 2x, 33% at 5x, rest at 10x" | tiered_take_profit |
| "hold for max N hours/bars" | time_exit (in exit.other[]) |
| "exit when indicator flips" | indicator_reversal (in exit.other[]) |
| "exit when smart money sells" | smart_money_sell (in exit.other[]) |
| "bail if dev dumps" | dev_dump (in exit.other[]) |
| "mirror their sells" | wallet_mirror_sell (in exit.other[]) |
| "bail if price crashes fast" | fast_dump_exit (in exit.other[]) |
Sizing selection
| User says... | Use |
|---|
| "$100 per trade", "fixed amount" | fixed_usd |
| "2% of portfolio per trade" | fixed_pct |
| "size based on volatility", "risk parity" | volatility_scaled |
Critical distinction: smart_money_buy vs smart_money_present_min (G-05)
smart_money_buy (E-07, entry trigger): Event-driven. Fires when a SM wallet executes a buy transaction. Use when the user wants to react to SM activity.
smart_money_present_min (TF-11, filter): State check at entry-candidate time. Checks how many SM wallets currently hold the token. Use when the user wants to confirm SM presence before entering on a different trigger.
- They can be combined:
smart_money_buy as entry + smart_money_present_min as filter (require >=2 SM holders AND react to a new SM buy).
4. Harness Rules
The harness validates every spec before execution. Violations are rejected with a plain-English error — fix and regenerate.
| Rule | Name | Enforcement |
|---|
| H-01 | No missing stop_loss | exit.stop_loss.pct is required. A strategy without a stop is gambling. |
| H-02 | No martingale | Rejects specs that increase size after a loss or re-enter losing positions at lower prices. |
| H-03 | SL must be tighter than TP | If stop_loss.pct >= take_profit.pct, negative asymmetry. Rejected. Does NOT apply when using tiered_take_profit or trailing_stop instead of take_profit. |
| H-04 | Param bounds respected | Every param must be within its schema-defined min/max range. |
| H-05 | Daily risk cap | fixed_pct * max_daily_trades.n must not exceed 20% equity per day. |
| H-06 | No unknown types | Every "type" field must reference a primitive in this library. No freeform code. |
5. Grammar Rules
| Rule | Topic | Resolution |
|---|
| G-01 | Exit semantics | All exits evaluated in parallel every tick. First-to-fire closes position. On same-tick tie, stop_loss wins (fail-safe). No priority ordering. |
| G-02 | Dynamic universe | When instrument.symbol is "*", the universe block is required with selector (which entry primitive produces the token set) and chain. |
| G-03 | take_profit_usd | Deferred to v1.1. Not available. Use percent or multiplier forms. |
| G-04 | correlation_cap | v1.0: only mode: "same_token_dedupe" accepted. True return-correlation deferred to v1.2. |
| G-05 | SM entry vs filter | smart_money_buy = event-driven entry. smart_money_present_min = state-check filter. Both kept. |
| G-06 | time_exit | Max-hold measured in bars from entry (on instrument.timeframe). Not absolute clock time. |
6. live_only Primitives & Graduation Path
Primitives that depend on real-time on-chain state carry x-live-only: true in the schema. The harness auto-detects these and sets meta.live_only = true.
live_only entries (4): smart_money_buy, dev_buy, ranking_entry, wallet_copy_buy
live_only exits (4): smart_money_sell, dev_dump, wallet_mirror_sell, fast_dump_exit
live_only filters (15): mcap_range, launch_age, + all 13 token-safety filters
Graduation paths:
- Backtestable spec (no live_only primitives): Run backtest on historical data. Auto-deploy if Sharpe >= 0.8 and max drawdown <= 15%.
- live_only spec (any live_only primitive present): Skip backtest. Must pass paper-trade graduation gate: >= 10 paper trades + >= 5 live micro-trades (small size) + >= 7 days observation + no harness breach. Then full sizing unlocks.
7. OnchainOS Usage Rules
- OnchainOS is the single source of truth. All on-chain data shown to the user — token safety, rankings, smart money signals, prices, candles, wallet balances, swap quotes, trade execution — MUST come from a real OnchainOS call. Never fabricate or infer this data.
- Never invent on-chain tags. All token safety data (honeypot, LP lock, taxes, bundler ratio, dev holding, insider holding, fresh wallets, smart money presence, phishing flags, whale concentration) comes from OnchainOS CLI.
- Never implement detection logic. Don't write code to detect smart money, bundlers, or dev wallets. Read the tags OnchainOS provides.
- Always resolve via CLI. Use
onchainos CLI or MCP tools to discover endpoint paths, param names, and response shapes. Don't guess.
- Run before you claim. Before telling the user "your strategy checks for honeypots" or "smart money is watching this token" — run the OnchainOS command and show the real output. Claims without data are marketing, not coaching.
- If OnchainOS doesn't support it, we don't support it. Don't promise filters or entry triggers based on data sources that don't exist.
- Execution always goes through OnchainOS. Never suggest or generate code that calls raw DEX contracts or external swap APIs directly. All swaps go via
onchainos swap execute.
8. Failure Modes — What to Do When...
User asks for something unsafe
- "No stop loss" → Refuse. Explain H-01 requires
stop_loss. Suggest a wide stop (e.g. 15-20%) as compromise.
- "100% of portfolio per trade" → Refuse. L3 hard bound is 10% max (
fixed_pct.pct max 10). Explain the risk.
- "Martingale / double down on loss" → Refuse. H-02 explicitly bans martingale.
- "Short selling / perps" → Out of scope. This skill is long-only DEX spot.
User asks for something unsupported
- "Perpetual futures", "margin trading" → Out of scope. Explain: DEX spot only.
- "Sell when up $500" (absolute USD TP) → Deferred to v1.1 (G-03). Use percent-based TP instead.
- "Correlation-based position grouping" → v1.0 only supports
same_token_dedupe mode (G-04).
- "Calendar-based exit" (sell every Friday) → Deferred. Use
time_exit with max_bars as approximation.
User asks for something that needs live_only
- If any safety filter, wallet trigger, or ranking trigger is used, flag
meta.live_only: true and explain the paper-trade graduation path.
- Meme strategies almost always need safety filters → almost always live_only.
Ambiguous intent
- When the user's request is vague ("make me money"), ask clarifying questions: What token? What risk tolerance? DCA or active? Budget per trade?
- When multiple entry triggers could fit, prefer the simplest one that matches intent.
9. Worked Examples
These 5 examples show the complete translation from user prompt → JSON spec. All param names match schema.json (the Primitive Library is source of truth).
Example 1: SOL Dip Buyer — US Hours Only
User prompt:
"Buy SOL whenever it drops 5% in the last hour, but only during US trading hours (9:30am-4pm ET), max 3 buys per day, $200 per buy, stop out at 8%, take profit at 10%. Pause the strategy if my week is down more than 10%."
Reasoning:
- 5% drop in 1h on 5m bars →
price_drop with pct: 5, lookback_bars: 12 (12 five-minute bars = 1 hour)
- US trading hours →
time_window filter with start_hour: 13, end_hour: 20 (UTC, covers 9:30-4pm ET approximately), weekdays_only: true
- Max 3 buys/day →
max_daily_trades risk overlay
- $200 per buy →
fixed_usd
- Stop 8%, TP 10% →
stop_loss + take_profit
- Week down 10% →
drawdown_pause with pause_pct: 10
- Add
cooldown filter (6 bars = 30 min on 5m timeframe) so rapid-fire dips don't exhaust budget
Spec:
{
"meta": {
"name": "sol_dip_us_hours",
"version": "1.0",
"risk_tier": "conservative",
"description": "Buy SOL on 5% hourly dips during US trading hours",
"author_intent": "Buy SOL whenever it drops 5% in the last hour, but only during US trading hours, max 3 buys per day, $200 per buy, stop out at 8%, take profit at 10%. Pause if my week is down more than 10%."
},
"instrument": {
"symbol": "SOL-USDC",
"timeframe": "5m"
},
"entry": {
"type": "price_drop",
"pct": 5,
"lookback_bars": 12
},
"exit": {
"stop_loss": { "pct": 8 },
"take_profit": { "pct": 10 }
},
"sizing": {
"type": "fixed_usd",
"usd": 200
},
"filters": [
{ "type": "time_window", "start_hour": 13, "end_hour": 20, "weekdays_only": true },
{ "type": "cooldown", "bars": 6 }
],
"risk_overlays": [
{ "type": "max_daily_trades", "n": 3 },
{ "type": "drawdown_pause", "pause_pct": 10 }
]
}
Graduation: Backtestable. All primitives are price/time based. Run on 12 months of SOL 5m bars.
Example 2: BTC Weekly DCA
User prompt:
"DCA $100 into BTC every Monday at 9am UTC. No exit — I'm holding. But add a 20% trailing stop just so a flash-crash below my avg cost doesn't wreck me."
Reasoning:
- Weekly DCA →
time_schedule with interval: "1W", anchor_utc: "09:00"
- $100 flat →
fixed_usd
- "No exit" but user asked for trailing stop →
trailing_stop at 20%. Note: trailing_stop max is 20, fits exactly.
- H-01 requires
stop_loss → add stop_loss at 20% as backstop (same threshold as trailing, so trailing fires first in practice)
- No filters or risk overlays needed — DCA is intentionally simple.
Spec:
{
"meta": {
"name": "btc_weekly_dca",
"version": "1.0",
"risk_tier": "passive",
"description": "Weekly DCA into BTC with trailing stop safety net",
"author_intent": "DCA $100 into BTC every Monday at 9am UTC. No exit, but add a 20% trailing stop for flash-crash protection."
},
"instrument": {
"symbol": "WBTC-USDC",
"timeframe": "1D"
},
"entry": {
"type": "time_schedule",
"interval": "1W",
"anchor_utc": "09:00"
},
"exit": {
"stop_loss": { "pct": 20 },
"trailing_stop": { "pct": 20 }
},
"sizing": {
"type": "fixed_usd",
"usd": 100
},
"filters": [],
"risk_overlays": []
}
Graduation: Backtestable. Deterministic schedule + price-based exits. Run on 24 months of BTC daily bars.
Example 3: Meme Safety-First — Full Safety Stack
User prompt:
"I want to snipe new meme coins ranked in the top 20 trending but I don't want to get rugged. Check everything — honeypot, LP burns, taxes under 5%, no bundler pumps, dev holding under 10%, at least one smart money already in. Risk $50 per trade, tiered take-profit at 2x/5x/10x, hard stop at -50%."
Reasoning:
- Top 20 trending →
ranking_entry with list_name: "trending", top_n: 20
- Dynamic universe →
symbol: "*", needs universe block
- "Check everything" → all 13 safety filters
- Taxes under 5% → separate
buy_tax_max + sell_tax_max at max_pct: 5
- "At least one smart money in" →
smart_money_present_min filter with min_wallets: 1
- Tiered TP at 2x/5x/10x →
tiered_take_profit with pct_gain values of 100/400/900 (2x = +100%, 5x = +400%, 10x = +900%)
- Hard stop -50% →
stop_loss.pct: 20 (capped at schema max of 20 — inform user)
- Implied:
mcap_range + launch_age for "new meme coin"
- All safety filters + ranking_entry →
live_only: true
Spec:
{
"meta": {
"name": "meme_safety_first",
"version": "1.0",
"risk_tier": "aggressive",
"live_only": true,
"description": "Snipe trending meme tokens with full safety filter stack",
"author_intent": "Snipe new meme coins ranked in the top 20 trending, check everything for safety, $50 per trade, tiered TP at 2x/5x/10x, hard stop at -50%."
},
"instrument": {
"symbol": "*",
"timeframe": "5m"
},
"universe": {
"selector": "ranking_entry",
"chain": "solana"
},
"entry": {
"type": "ranking_entry",
"list_name": "trending",
"top_n": 20
},
"exit": {
"stop_loss": { "pct": 20 },
"tiered_take_profit": {
"tiers": [
{ "pct_gain": 100, "pct_sell": 33 },
{ "pct_gain": 400, "pct_sell": 33 },
{ "pct_gain": 900, "pct_sell": 34 }
]
}
},
"sizing": {
"type": "fixed_usd",
"usd": 50
},
"filters": [
{ "type": "mcap_range", "min_usd": 100000, "max_usd": 5000000 },
{ "type": "launch_age", "min_hours": 2, "max_hours": 168 },
{ "type": "honeypot_check" },
{ "type": "lp_locked", "min_pct_locked": 80, "min_lock_days": 30 },
{ "type": "buy_tax_max", "max_pct": 5 },
{ "type": "sell_tax_max", "max_pct": 5 },
{ "type": "liquidity_min", "min_usd": 25000 },
{ "type": "top_holders_max", "top_n": 10, "max_pct": 35 },
{ "type": "bundler_ratio_max", "max_pct": 10 },
{ "type": "dev_holding_max", "max_pct": 10 },
{ "type": "insider_holding_max", "max_pct": 15 },
{ "type": "fresh_wallet_ratio_max", "max_pct": 25 },
{ "type": "smart_money_present_min", "min_wallets": 1 },
{ "type": "phishing_exclude" },
{ "type": "whale_concentration_max", "max_pct": 20 }
],
"risk_overlays": [
{ "type": "max_concurrent_positions", "n": 5 },
{ "type": "max_daily_trades", "n": 10 },
{ "type": "session_loss_pause", "max_consecutive_losses": 3 }
]
}
Note: User asked for -50% stop but stop_loss.pct max is 20. Inform the user: "Schema enforces a maximum 20% stop loss for safety. Your position will be stopped at -20% instead of -50%."
Graduation: live_only. Paper gate: >= 10 paper trades + >= 5 live micro-trades at $5 + >= 7 days observation, then $50 sizing unlocks.
Example 4: Smart Money Copy-Trade
User prompt:
"Copy-trade these 3 wallets on Base: 0xabc..., 0xdef..., 0x123.... When any of them buys a token, I buy the same token with 2% of my portfolio. Mirror their sells too. Also bail immediately if the dev dumps, or if liquidity drops under $50k. Pause the whole thing if I'm down more than 15% this week."
Reasoning:
- Named wallets →
wallet_copy_buy with target_wallet as array
- Mirror sells →
wallet_mirror_sell in exit.other[]
- "Bail if dev dumps" →
dev_dump in exit.other[]
- "Bail if price crashes" (liquidity proxy) →
fast_dump_exit in exit.other[]
- Pre-entry liquidity gate →
liquidity_min filter
- 2% of portfolio →
fixed_pct
- Week down 15% →
drawdown_pause with pause_pct: 15
- Add
honeypot_check + phishing_exclude — copy-trading without these is reckless
- Dynamic universe →
symbol: "*" + universe block
Spec:
{
"meta": {
"name": "smart_money_copy",
"version": "1.0",
"risk_tier": "moderate",
"live_only": true,
"description": "Copy-trade 3 wallets on Base with safety exits",
"author_intent": "Copy-trade 3 wallets on Base, 2% portfolio per trade, mirror sells, bail on dev dump or liquidity drop, pause at 15% weekly drawdown."
},
"instrument": {
"symbol": "*",
"timeframe": "5m"
},
"universe": {
"selector": "wallet_copy_buy",
"chain": "base"
},
"entry": {
"type": "wallet_copy_buy",
"target_wallet": [
"0xabc0000000000000000000000000000000000abc",
"0xdef0000000000000000000000000000000000def",
"0x1230000000000000000000000000000000000123"
],
"min_usd": 100,
"mirror_mode": "instant"
},
"exit": {
"stop_loss": { "pct": 15 },
"other": [
{
"type": "wallet_mirror_sell",
"target_wallet": [
"0xabc0000000000000000000000000000000000abc",
"0xdef0000000000000000000000000000000000def",
"0x1230000000000000000000000000000000000123"
],
"min_pct_sold": 50
},
{ "type": "dev_dump", "min_usd": 500 },
{ "type": "fast_dump_exit", "drop_pct": 30, "window_sec": 60 }
]
},
"sizing": {
"type": "fixed_pct",
"pct": 2
},
"filters": [
{ "type": "liquidity_min", "min_usd": 50000 },
{ "type": "honeypot_check" },
{ "type": "phishing_exclude" }
],
"risk_overlays": [
{ "type": "drawdown_pause", "pause_pct": 15 },
{ "type": "max_concurrent_positions", "n": 8 },
{ "type": "correlation_cap", "mode": "same_token_dedupe", "max_correlated": 3 }
]
}
Graduation: live_only. Wallet activity can't be replayed. Paper gate: >= 10 paper + >= 5 live micro + >= 7 days.
Example 5: Launchpad Sniper
User prompt:
"Snipe brand-new tokens on the OKX launchpad — only tokens launched in the last 48 hours. Require at least 2 smart-money wallets to be in already, no bundlers over 5%, LP must be locked, mcap between $50k and $2M. $75 per trade, max 3 positions at once. Take profits at 1.5x / 3x / 6x. Bail if price drops more than 40% in 5 minutes."
Reasoning:
- "Brand-new tokens on launchpad" →
ranking_entry with list_name: "new", top_n: 50
- "Last 48 hours" →
launch_age filter with max_hours: 48
- "2 smart money in" →
smart_money_present_min with min_wallets: 2
- "No bundlers over 5%" →
bundler_ratio_max with max_pct: 5
- "LP locked" →
lp_locked with min_pct_locked: 80
- "Mcap $50k-$2M" →
mcap_range
- $75/trade →
fixed_usd
- Max 3 positions →
max_concurrent_positions
- TP at 1.5x/3x/6x →
tiered_take_profit with pct_gain 50/200/500
- "Bail if drops 40% in 5 min" →
fast_dump_exit with drop_pct: 40, window_sec: 300
- Add
stop_loss backstop at 20% so failed snipes don't bleed forever
- Add
honeypot_check + phishing_exclude as baseline safety
Spec:
{
"meta": {
"name": "launchpad_sniper",
"version": "1.0",
"risk_tier": "speculative",
"live_only": true,
"description": "Snipe new tokens under 48h with SM confirmation and safety stack",
"author_intent": "Snipe brand-new tokens, last 48 hours, require 2 SM wallets, no bundlers over 5%, LP locked, mcap $50k-$2M, $75/trade, max 3 positions, TP at 1.5x/3x/6x, bail on 40% drop in 5 min."
},
"instrument": {
"symbol": "*",
"timeframe": "1m"
},
"universe": {
"selector": "ranking_entry",
"chain": "solana"
},
"entry": {
"type": "ranking_entry",
"list_name": "new",
"top_n": 50
},
"exit": {
"stop_loss": { "pct": 20 },
"tiered_take_profit": {
"tiers": [
{ "pct_gain": 50, "pct_sell": 40 },
{ "pct_gain": 200, "pct_sell": 30 },
{ "pct_gain": 500, "pct_sell": 30 }
]
},
"other": [
{ "type": "fast_dump_exit", "drop_pct": 40, "window_sec": 300 }
]
},
"sizing": {
"type": "fixed_usd",
"usd": 75
},
"filters": [
{ "type": "launch_age", "max_hours": 48 },
{ "type": "mcap_range", "min_usd": 50000, "max_usd": 2000000 },
{ "type": "lp_locked", "min_pct_locked": 80, "min_lock_days": 30 },
{ "type": "bundler_ratio_max", "max_pct": 5 },
{ "type": "smart_money_present_min", "min_wallets": 2 },
{ "type": "honeypot_check" },
{ "type": "phishing_exclude" }
],
"risk_overlays": [
{ "type": "max_concurrent_positions", "n": 3 },
{ "type": "session_loss_pause", "max_consecutive_losses": 3 }
]
}
Graduation: live_only. Paper gate: >= 10 paper + >= 5 live micro-trades at $10 + >= 7 days, then $75 sizing unlocks.
10. Example Coverage Scorecard
| Category | Hit | Miss (unhit in examples, but available) |
|---|
| Entry (12) | price_drop, time_schedule, ranking_entry, wallet_copy_buy | price_breakout, ma_cross, rsi_threshold, volume_spike, smart_money_buy, dev_buy, macd_cross, bollinger_touch |
| Exit (10) | stop_loss, take_profit, trailing_stop, tiered_take_profit, wallet_mirror_sell, dev_dump, fast_dump_exit | time_exit, indicator_reversal, smart_money_sell |
| Filter (23) | time_window, cooldown, mcap_range, launch_age + all 13 safety | volatility_range, volume_minimum, market_regime, price_range, btc_overlay, top_zone_guard |
| Sizing (3) | fixed_usd, fixed_pct | volatility_scaled |
| Risk (5) | max_daily_trades, max_concurrent_positions, drawdown_pause, correlation_cap, session_loss_pause | (all covered) |
The 5 examples cover 33/53 primitives. The remaining 20 are straightforward — refer to the primitive tables above for their exact param names and ranges.