Files
nofx/kernel/engine_prompt_test.go
tinkle-community 110bf52908 feat: cream terminal redesign, English-only UI, autopilot launch fixes
- 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
2026-06-30 16:03:52 +08:00

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
}