mirror of
https://github.com/NoFxAiOS/nofx.git
synced 2026-07-03 02:50:59 +08:00
- Redesign dashboard into a cream-paper + vermilion IBM Plex Mono terminal (live L2 order book, cost/liq map, WS K-line, signal matrix, orchestration topology, risk radar, execution log, current positions, equity curve) - Convert all user-facing UI and backend strings/prompts from Chinese to English (multi-language retained, default English) - Add /api/statistics/full endpoint + full-stats frontend wiring - Fix Autopilot launch: reuse the existing trader instead of creating duplicates (eliminates repeat ~35s create cost and stale-trader 404s); launch sends 5m scan interval - Fix unreadable toasts: cream theme with high-contrast text + per-type accent - Silence background dashboard polls (getTraderConfig) to stop error-toast spam
125 lines
3.9 KiB
Go
125 lines
3.9 KiB
Go
package kernel
|
|
|
|
import (
|
|
"strings"
|
|
"testing"
|
|
|
|
"nofx/store"
|
|
)
|
|
|
|
func TestBuildSystemPromptUsesVergexClaw402Prompt(t *testing.T) {
|
|
cfg := store.GetDefaultStrategyConfig("zh")
|
|
cfg.CoinSource.SourceType = "vergex_signal"
|
|
cfg.CoinSource.VergexLimit = 5
|
|
cfg.PromptSections.RoleDefinition = "# You are a professional Hyperliquid USDC multi-asset trading AI"
|
|
cfg.CustomPrompt = "Long only, no shorts."
|
|
|
|
engine := NewStrategyEngine(&cfg)
|
|
prompt := engine.BuildSystemPrompt(30, "balanced")
|
|
|
|
if !strings.Contains(prompt, "NOFX Claw402 auto-trader") {
|
|
t.Fatalf("prompt did not use the Claw402/Vergex TradeFi role:\n%s", prompt)
|
|
}
|
|
if !strings.Contains(prompt, "Claw402.ai Signal Ranking") || !strings.Contains(prompt, "Signal Lab") || !strings.Contains(prompt, "Cost/Liquidation Heatmap") {
|
|
t.Fatalf("prompt is missing Claw402/Vergex detail data guidance:\n%s", prompt)
|
|
}
|
|
if !strings.Contains(prompt, "open_short") {
|
|
t.Fatalf("prompt should explicitly allow short entries:\n%s", prompt)
|
|
}
|
|
if !strings.Contains(prompt, "Direction must be data-driven") {
|
|
t.Fatalf("prompt should explain that direction is data-driven, not long-only:\n%s", prompt)
|
|
}
|
|
if !strings.Contains(prompt, "every open position must use exactly 10x") {
|
|
t.Fatalf("prompt should force 10x leverage for Claw402 opens:\n%s", prompt)
|
|
}
|
|
if !strings.Contains(prompt, "use the full max notional per position") {
|
|
t.Fatalf("prompt should force full-size Claw402 opens:\n%s", prompt)
|
|
}
|
|
if containsCJK(prompt) {
|
|
t.Fatalf("system prompt must be English-only, got CJK text:\n%s", prompt)
|
|
}
|
|
legacyPhrases := []string{
|
|
"Hyperliquid USDC multi-asset trading AI",
|
|
"Long only",
|
|
"Altcoin",
|
|
"BTC/ETH",
|
|
"LONG-ONLY",
|
|
"Do not short",
|
|
"MUST open a long",
|
|
}
|
|
for _, phrase := range legacyPhrases {
|
|
if strings.Contains(prompt, phrase) {
|
|
t.Fatalf("prompt still contains legacy phrase %q:\n%s", phrase, prompt)
|
|
}
|
|
}
|
|
}
|
|
|
|
func TestBuildSystemPromptFallsBackToEnglishWhenConfiguredLanguageIsChinese(t *testing.T) {
|
|
cfg := store.GetDefaultStrategyConfig("zh")
|
|
cfg.CoinSource.SourceType = "static"
|
|
cfg.CoinSource.StaticCoins = []string{"BTCUSDT", "ETHUSDT"}
|
|
cfg.CoinSource.VergexLimit = 0
|
|
cfg.CoinSource.VergexMarketType = ""
|
|
cfg.CoinSource.VergexChain = ""
|
|
cfg.PromptSections.RoleDefinition = "# You are a Chinese system prompt"
|
|
cfg.PromptSections.TradingFrequency = "# High-frequency trading\nTrade every minute."
|
|
cfg.PromptSections.EntryStandards = "# Entry\nOpen positions freely."
|
|
cfg.PromptSections.DecisionProcess = "# Decision\nOutput directly."
|
|
cfg.CustomPrompt = "Chinese preference should not enter the system prompt."
|
|
|
|
engine := NewStrategyEngine(&cfg)
|
|
prompt := engine.BuildSystemPrompt(30, "balanced")
|
|
|
|
required := []string{
|
|
"Data Dictionary & Trading Rules",
|
|
"You are a professional Hyperliquid USDC multi-asset trading AI",
|
|
"Trading Frequency Awareness",
|
|
"Entry Standards",
|
|
"Decision Process",
|
|
}
|
|
for _, phrase := range required {
|
|
if !strings.Contains(prompt, phrase) {
|
|
t.Fatalf("English fallback prompt missing %q:\n%s", phrase, prompt)
|
|
}
|
|
}
|
|
if containsCJK(prompt) {
|
|
t.Fatalf("system prompt must be English-only, got CJK text:\n%s", prompt)
|
|
}
|
|
}
|
|
|
|
func TestBuildSystemPromptDoesNotForceLongOnlyForSingleXYZ(t *testing.T) {
|
|
prompt := buildXYZStockCustomPrompt("XYZ:INTC")
|
|
|
|
required := []string{
|
|
"DIRECTIONAL, SIGNAL-DRIVEN",
|
|
"You may open long or short",
|
|
"open_short",
|
|
}
|
|
for _, phrase := range required {
|
|
if !strings.Contains(prompt, phrase) {
|
|
t.Fatalf("single XYZ prompt missing %q:\n%s", phrase, prompt)
|
|
}
|
|
}
|
|
|
|
forbidden := []string{
|
|
"LONG-ONLY",
|
|
"Do not short",
|
|
"MUST open a long",
|
|
"Probing > waiting",
|
|
}
|
|
for _, phrase := range forbidden {
|
|
if strings.Contains(prompt, phrase) {
|
|
t.Fatalf("single XYZ prompt still contains forced-long phrase %q:\n%s", phrase, prompt)
|
|
}
|
|
}
|
|
}
|
|
|
|
func containsCJK(text string) bool {
|
|
for _, r := range text {
|
|
if r >= 0x4E00 && r <= 0x9FFF {
|
|
return true
|
|
}
|
|
}
|
|
return false
|
|
}
|