feat(i18n): bilingual EN/ZH setup guide with language selection

store/telegram_config.go:
- Add Language field to TelegramConfig (persisted in DB)
- Add SetLanguage(lang) and GetLanguage() methods
- Default language: English (en)

telegram/bot.go:
- First /start triggers language selection (1=English, 2=中文)
- /lang command to change language at any time
- awaitingLang state machine handles language choice before any other input
- buildSetupGuide() now fully bilingual (EN/ZH), context-aware:
  Step 1: configure AI model (no model yet)
  Step 2: configure exchange (model OK, no exchange)
  Ready: show full capabilities
- tryHandleSetupCommand() bilingual: 'configure/配置 <provider> <key>'
- helpMessage(lang) fully bilingual
- All error/status messages bilingual

Default: English. isLangDefault() detects whether user has explicitly
chosen a language vs falling back to the 'en' default.
This commit is contained in:
tinkle-community
2026-03-08 18:44:38 +08:00
parent 1bbd4b44ac
commit d160301359
2 changed files with 242 additions and 55 deletions

View File

@@ -17,6 +17,7 @@ type TelegramConfig struct {
Username string `gorm:"column:username"` // @username for display
BoundAt time.Time `gorm:"column:bound_at"`
ModelID string `gorm:"column:model_id;default:''"` // AI model used for Telegram replies
Language string `gorm:"column:language;default:''"` // "zh" or "en"; empty = not chosen yet
CreatedAt time.Time
UpdatedAt time.Time
}
@@ -40,6 +41,8 @@ type TelegramConfigStore interface {
IsBound() (bool, error) // Check if any user is bound
GetBoundChatID() (int64, error) // Get bound chat ID (0 if not bound)
Unbind() error // Remove binding
SetLanguage(lang string) error // Set UI language ("en" or "zh")
GetLanguage() string // Get UI language; returns "en" if not set
}
type telegramConfigStore struct {
@@ -133,3 +136,29 @@ func (s *telegramConfigStore) Unbind() error {
"username": "",
}).Error
}
func (s *telegramConfigStore) SetLanguage(lang string) error {
s.mu.Lock()
defer s.mu.Unlock()
var cfg TelegramConfig
result := s.db.First(&cfg, 1)
if result.Error != nil && !errors.Is(result.Error, gorm.ErrRecordNotFound) {
return result.Error
}
cfg.ID = 1
cfg.Language = lang
return s.db.Save(&cfg).Error
}
func (s *telegramConfigStore) GetLanguage() string {
s.mu.RLock()
defer s.mu.RUnlock()
var cfg TelegramConfig
if err := s.db.First(&cfg, 1).Error; err != nil {
return "en" // default: English
}
if cfg.Language == "" {
return "en"
}
return cfg.Language
}

View File

@@ -104,6 +104,10 @@ func runBot(token string, cfg *config.Config, st *store.Store) bool {
u.Timeout = 60
updates := bot.GetUpdatesChan(u)
// awaitingLang is true when the bot is waiting for the user to pick a language (1 or 2).
// It resets to false once a valid choice is received or /lang is re-issued.
awaitingLang := false
for update := range updates {
if update.Message == nil {
continue
@@ -111,56 +115,86 @@ func runBot(token string, cfg *config.Config, st *store.Store) bool {
chatID := update.Message.Chat.ID
text := update.Message.Text
// Handle /start: auto-bind or welcome with setup guide
// Language selection state: user must choose "1" or "2" after /start on first use.
// awaitingLang is true only until the user makes a choice (or we fall back to "en").
if awaitingLang && chatID == allowedChatID {
lang := parseLangChoice(text)
if lang != "" {
awaitingLang = false
st.TelegramConfig().SetLanguage(lang) //nolint:errcheck
sendMsg(bot, chatID, buildSetupGuide(st, botUserID, cfg.APIServerPort, botToken, lang))
} else {
sendMsg(bot, chatID, langSelectionMsg())
}
continue
}
// Handle /start: auto-bind or language selection / welcome
if text == "/start" {
if allowedChatID == 0 {
username := update.Message.From.UserName
if err := st.TelegramConfig().BindUser(chatID, "@"+username); err != nil {
logger.Errorf("Failed to bind Telegram user: %v", err)
sendMsg(bot, chatID, "绑定失败,请查看服务器日志。")
sendMsg(bot, chatID, "Binding failed. / 绑定失败。")
continue
}
allowedChatID = chatID
logger.Infof("Telegram bound to @%s (chatID: %d)", username, chatID)
} else if chatID != allowedChatID {
sendMsg(bot, chatID, "该机器人已被其他用户绑定。")
sendMsg(bot, chatID, "This bot is already bound to another user. / 该机器人已被其他用户绑定。")
continue
} else {
agents.Reset(chatID)
}
sendMsg(bot, chatID, buildSetupGuide(st, botUserID, cfg.APIServerPort, botToken))
// Show language selection if not chosen yet; otherwise go straight to guide.
lang := st.TelegramConfig().GetLanguage()
if lang == "en" && isLangDefault(st) {
// First time: ask language preference
awaitingLang = true
sendMsg(bot, chatID, langSelectionMsg())
} else {
sendMsg(bot, chatID, buildSetupGuide(st, botUserID, cfg.APIServerPort, botToken, lang))
}
continue
}
// Handle /lang: change language at any time
if text == "/lang" {
awaitingLang = true
sendMsg(bot, chatID, langSelectionMsg())
continue
}
// Handle /help
if text == "/help" {
sendMsg(bot, chatID, helpMessage())
lang := st.TelegramConfig().GetLanguage()
sendMsg(bot, chatID, helpMessage(lang))
continue
}
// Access control
if allowedChatID != 0 && chatID != allowedChatID {
sendMsg(bot, chatID, "无权限访问。")
sendMsg(bot, chatID, "Unauthorized. / 无权限访问。")
continue
}
if allowedChatID == 0 {
sendMsg(bot, chatID, "请先发送 /start 绑定账号。")
sendMsg(bot, chatID, "Send /start first. / 请先发送 /start。")
continue
}
if text == "" {
continue
}
// Direct setup commands — handled without LLM so they work even before
// an AI model is configured. Format: "配置 <provider> <api-key>"
if reply, handled := tryHandleSetupCommand(text, cfg.APIServerPort, botToken, st, botUserID); handled {
// Direct setup commands (no LLM needed): "configure deepseek sk-xxx" / "配置 deepseek sk-xxx"
lang := st.TelegramConfig().GetLanguage()
if reply, handled := tryHandleSetupCommand(text, cfg.APIServerPort, botToken, st, botUserID, lang); handled {
sendMsg(bot, chatID, reply)
continue
}
// Check if AI model is configured before entering agent loop.
// Guard: if no AI model configured, show setup guide instead of failing.
if newLLMClient(st, botUserID) == nil {
sendMsg(bot, chatID, buildSetupGuide(st, botUserID, cfg.APIServerPort, botToken))
sendMsg(bot, chatID, buildSetupGuide(st, botUserID, cfg.APIServerPort, botToken, lang))
continue
}
@@ -286,18 +320,53 @@ func clientForProvider(provider string) mcp.AIClient {
}
}
// buildSetupGuide checks the current configuration state and returns a contextual
// onboarding message. Called on every /start so the user always knows what to do next.
func buildSetupGuide(st *store.Store, userID string, apiPort int, botToken string) string {
// ── Language selection ────────────────────────────────────────────────────────
// langSelectionMsg is always bilingual so it works before a language is chosen.
func langSelectionMsg() string {
return `🌐 *Please select your language / 请选择语言*
1⃣ English
2⃣ 中文
Reply with 1 or 2 / 发送 1 或 2`
}
// parseLangChoice returns "en", "zh", or "" (unrecognised).
func parseLangChoice(text string) string {
switch strings.TrimSpace(text) {
case "1", "English", "english", "en", "EN":
return "en"
case "2", "中文", "zh", "ZH", "chinese", "Chinese":
return "zh"
}
return ""
}
// isLangDefault returns true if the user has never explicitly picked a language
// (i.e. the stored value is empty — the "en" default from GetLanguage() is a fallback).
func isLangDefault(st *store.Store) bool {
cfg, err := st.TelegramConfig().Get()
if err != nil {
return true
}
return cfg.Language == ""
}
// ── Setup guide ───────────────────────────────────────────────────────────────
// buildSetupGuide returns a context-aware onboarding message in the chosen language.
func buildSetupGuide(st *store.Store, userID string, apiPort int, botToken string, lang string) string {
// Step 1: AI model configured?
if _, err := st.AIModel().GetDefault(userID); err != nil {
return `🤖 *NOFX 个人 AI 交易助手*
if lang == "zh" {
return `🤖 *NOFX 个人 AI 交易助手*
欢迎!开始交易前,需要配置 AI 模型。
欢迎!开始需要配置 AI 模型。
*第一步:配置 AI 模型*
选择你有账号的服务商,发送以下格式的消息
发送以下格式(选一个你有账号的服务商)
` + "```" + `
配置 deepseek 你的API-Key
@@ -309,8 +378,33 @@ func buildSetupGuide(st *store.Store, userID string, apiPort int, botToken strin
配置 gemini 你的API-Key
` + "```" + `
*推荐*DeepSeek 价格最低,效果
获取 Keyhttps://platform.deepseek.com/api_keys`
*推荐*DeepSeek 价格低、效果好
获取 Keyhttps://platform.deepseek.com/api_keys
发送 /lang 切换语言`
}
return `🤖 *NOFX Personal AI Trading Bot*
Welcome! You need to configure an AI model before trading.
*Step 1: Configure AI Model*
Send a message in this format (pick a provider you have access to):
` + "```" + `
configure deepseek your-api-key
configure openai your-api-key
configure claude your-api-key
configure qwen your-api-key
configure kimi your-api-key
configure grok your-api-key
configure gemini your-api-key
` + "```" + `
*Recommended*: DeepSeek — low cost, great performance
Get your key: https://platform.deepseek.com/api_keys
Send /lang to change language`
}
// Step 2: Exchange configured?
@@ -323,25 +417,37 @@ func buildSetupGuide(st *store.Store, userID string, apiPort int, botToken strin
}
}
if !hasEnabled {
return `✅ AI 模型已配置!
if lang == "zh" {
return `✅ AI 模型已配置!
*第二步:配置交易所*
直接发消息告诉我交易所信息,例如:
_"帮我配置 OKX 交易所API Key 是 xxxSecret 是 xxxPassphrase 是 xxx"_
_"帮我配置 OKXAPI Key 是 xxxSecret 是 xxxPassphrase 是 xxx"_
_"帮我配置 BinanceAPI Key 是 xxxSecret Key 是 xxx"_
_"帮我配置 BybitAPI Key 是 xxxSecret Key 是 xxx"_
交易所 API Key 去交易所官网 → 账户设置 → API 管理 → 新建 Key(需开启合约交易权限)`
去交易所官网 → 账户设置 → API 管理 → 新建(需开启合约交易权限)`
}
return `✅ AI model configured!
*Step 2: Configure Exchange*
Just tell me your exchange credentials, for example:
_"Configure OKX, API Key is xxx, Secret is xxx, Passphrase is xxx"_
_"Configure Binance, API Key is xxx, Secret Key is xxx"_
_"Configure Bybit, API Key is xxx, Secret Key is xxx"_
Go to your exchange → Account → API Management → Create Key (enable futures/contract trading)`
}
// All configured — show full capabilities
return `✅ *NOFX 交易助手已就绪*
// All configured
if lang == "zh" {
return `✅ *NOFX 交易助手已就绪*
直接用自然语言告诉我你要做什么:
直接发消息告诉我你要做什么:
*查询*
_"查看我的持仓"_、_"查看账户余额"_
@@ -353,24 +459,42 @@ _"创建保守型策略,只交易 BTC 和 ETH"_
*控制*
_"启动交易员"_、_"暂停交易员"_
*查看数据*
_"查看今天的交易记录"_、_"查看盈亏统计"_
/start 重置对话 | /help 帮助 | /lang 切换语言`
}
return `✅ *NOFX Trading Bot Ready*
发送 /start 重置对话 | /help 查看更多`
Just tell me what you want to do:
*Query*
_"Show my positions"_, _"Show account balance"_
*Create & Start Trading*
_"Create a BTC trend strategy and start it"_
_"Create a conservative strategy, BTC and ETH only"_
*Control*
_"Start trader"_, _"Stop trader"_
/start reset session | /help | /lang change language`
}
// tryHandleSetupCommand handles "配置 <provider> <api-key>" commands directly
// without going through the LLM. This allows AI model setup even before any
// model is configured (bootstrapping problem).
func tryHandleSetupCommand(text string, apiPort int, botToken string, st *store.Store, userID string) (string, bool) {
// ── Direct setup commands (no LLM required) ───────────────────────────────────
// tryHandleSetupCommand intercepts "configure/配置 <provider> <key>" messages
// and calls PUT /api/models directly — no LLM needed, works during bootstrapping.
func tryHandleSetupCommand(text string, apiPort int, botToken string, st *store.Store, userID string, lang string) (string, bool) {
text = strings.TrimSpace(text)
if !strings.HasPrefix(text, "配置 ") && !strings.HasPrefix(strings.ToLower(text), "setup ") {
lower := strings.ToLower(text)
if !strings.HasPrefix(text, "配置 ") && !strings.HasPrefix(lower, "configure ") {
return "", false
}
parts := strings.Fields(text)
if len(parts) < 3 {
return "格式:配置 <服务商> <API-Key>\n例如配置 deepseek sk-xxxxxxxxx", true
if lang == "zh" {
return "格式:配置 <服务商> <API-Key>\n例如配置 deepseek sk-xxxxxxxxx", true
}
return "Format: configure <provider> <api-key>\nExample: configure deepseek sk-xxxxxxxxx", true
}
provider := strings.ToLower(parts[1])
@@ -381,49 +505,60 @@ func tryHandleSetupCommand(text string, apiPort int, botToken string, st *store.
"qwen": true, "kimi": true, "grok": true, "gemini": true,
}
if !validProviders[provider] {
return fmt.Sprintf("不支持的服务商:%s\n支持openai / deepseek / claude / qwen / kimi / grok / gemini", provider), true
if lang == "zh" {
return fmt.Sprintf("不支持的服务商:%s\n支持openai / deepseek / claude / qwen / kimi / grok / gemini", provider), true
}
return fmt.Sprintf("Unknown provider: %s\nSupported: openai / deepseek / claude / qwen / kimi / grok / gemini", provider), true
}
// Call PUT /api/models directly without LLM.
body, _ := json.Marshal(map[string]any{
"models": map[string]any{
provider: map[string]any{
"enabled": true,
"api_key": apiKey,
},
provider: map[string]any{"enabled": true, "api_key": apiKey},
},
})
req, err := http.NewRequest("PUT", fmt.Sprintf("http://127.0.0.1:%d/api/models", apiPort), bytes.NewReader(body))
if err != nil {
return "配置请求失败,请稍后重试", true
if lang == "zh" {
return "配置请求失败,请稍后重试", true
}
return "Failed to create request, please try again", true
}
req.Header.Set("Content-Type", "application/json")
req.Header.Set("Authorization", "Bearer "+botToken)
resp, err := (&http.Client{Timeout: 10 * time.Second}).Do(req)
if err != nil {
return "无法连接到服务,请确认服务正常运行", true
if lang == "zh" {
return "无法连接服务,请确认服务正常运行", true
}
return "Cannot reach service, please check it is running", true
}
defer resp.Body.Close()
respBody, _ := io.ReadAll(resp.Body)
if resp.StatusCode >= 400 {
return fmt.Sprintf("配置失败(%d%s", resp.StatusCode, string(respBody)), true
return fmt.Sprintf("Error %d: %s", resp.StatusCode, string(respBody)), true
}
logger.Infof("Bot: setup command configured provider=%s", provider)
return fmt.Sprintf("✅ %s 配置成功!\n\n发送 /start 查看下一步", provider), true
if lang == "zh" {
return fmt.Sprintf("✅ %s 配置成功!\n\n发送 /start 查看下一步", provider), true
}
return fmt.Sprintf("✅ %s configured successfully!\n\nSend /start to see the next step", provider), true
}
func helpMessage() string {
return `*NOFX 使用指南*
// ── Help message ──────────────────────────────────────────────────────────────
func helpMessage(lang string) string {
if lang == "zh" {
return `*NOFX 使用指南*
*查询*
- "查看我的持仓"
- "查看账户余额"
- "列出我的交易员"
*交易控制*
*控制*
- "启动交易员"
- "暂停 xxx 交易员"
@@ -431,13 +566,36 @@ func helpMessage() string {
- "帮我创建 BTC 趋势策略并跑起来"
- "创建保守型策略BTC ETH止损 8%"
*配置*(不需要 AI,直接发
- "配置 deepseek sk-xxxx"
- "帮我配置 OKX 交易所Key 是 xxx"
*直接配置(不需要 AI*
- 配置 deepseek sk-xxxx
- 配置 openai sk-xxxx
*命令*
/start - 重置对话 / 查看配置状态
/help - 显示此帮助
/lang - 切换语言
/help - 显示此帮助`
}
return `*NOFX Help*
支持中文和英文,直接说你想做什么就行。`
*Query*
- "Show my positions"
- "Show account balance"
- "List my traders"
*Control*
- "Start trader"
- "Stop trader [name]"
*Create strategy*
- "Create a BTC trend strategy and start it"
- "Create a conservative strategy, BTC and ETH, 8% stop loss"
*Direct setup (no AI needed)*
- configure deepseek sk-xxxx
- configure openai sk-xxxx
*Commands*
/start - Reset session / check setup status
/lang - Change language
/help - Show this help`
}