feat(auto-reply): 接入万川平台模型配置 + 各模型独立网关回退
万川平台对接 - 新增 wanchuan_proxy.go:WanchuanLogin/WanchuanGetModel 代理登录与按 code 拉取模型, 日志对 password/token/apiKey 打码(含 encryptedConfig 二次解析) - 新增 PlatformConfig(baseUrl/username/password)及 Get/SavePlatformConfig 持久化 - 前端万川卡片:登录→拉取 chat/vision/embedding/rerank/voice→回填 form 并保存→必要时重建向量索引 各模型独立网关(url+key),留空回退聊天网关 - RetrievalConfig 新增 embeddingBaseUrl/embeddingApiKey、rerankBaseUrl/rerankApiKey - embeddingRequestConfig/rerankRequestConfig:优先独立网关,未配置回退 AI.BaseURL/APIKey - vision/audio 同模式:非 DashScope 网关下视觉/语音模型留空时不再锁死或强写 DashScope, 运行期由 fallbackString(VisionModel, Model) 动态复用聊天模型 陈旧向量空间防护 - loadEmbeddingIndex 检测磁盘索引与当前 embedding 模型/维度不一致时清空向量、回退关键词检索, 并提示重建(embeddingIndexStaleReason,兼容旧版无模型名索引) UI 状态修复 - 登录拉模型期间统一置全局 busy,禁用闸门收敛为 busy(与刷新联系人等按钮同范式), platformBusy 仅保留用于按钮「处理中…」文案,杜绝并发读写 form 与反向可点洞 其他 - 删除遗留 helper/auto_reply_ai.go.bak - 补充 config/helper 单元测试(视觉回退分支、陈旧索引判定)
This commit is contained in:
@@ -99,3 +99,49 @@ func TestApplyDefaultsFixesWrongRerankConfig(t *testing.T) {
|
||||
t.Errorf("Expected rerank model to be corrected to 'qwen3-rerank', got %q", cfg.AutoReplyConfig.Retrieval.RerankModel)
|
||||
}
|
||||
}
|
||||
|
||||
// 非 DashScope 统一网关(如万川)且无独立视觉网关时:视觉模型 == 聊天模型应被清空,
|
||||
// 以便运行期 fallbackString(VisionModel, Model) 动态跟随聊天模型,不锁死旧值。
|
||||
func TestApplyDefaultsNonDashScopeVisionFollowsChat(t *testing.T) {
|
||||
cfg := NewDefaultConfig()
|
||||
cfg.AutoReplyConfig.AI.BaseURL = "https://wanchuan.example/v1"
|
||||
cfg.AutoReplyConfig.AI.Model = "wanchuan-chat"
|
||||
cfg.AutoReplyConfig.AI.VisionBaseURL = ""
|
||||
cfg.AutoReplyConfig.AI.VisionModel = "wanchuan-chat" // 与聊天模型相同(之前回填留下的值)
|
||||
|
||||
cfg.ApplyDefaults()
|
||||
|
||||
if cfg.AutoReplyConfig.AI.VisionModel != "" {
|
||||
t.Errorf("Expected vision model cleared to follow chat on non-DashScope unified gateway, got %q", cfg.AutoReplyConfig.AI.VisionModel)
|
||||
}
|
||||
}
|
||||
|
||||
// 非 DashScope 网关上用户在同一网关显式填了不同的视觉模型时,应保留其选择。
|
||||
func TestApplyDefaultsNonDashScopeKeepsExplicitVision(t *testing.T) {
|
||||
cfg := NewDefaultConfig()
|
||||
cfg.AutoReplyConfig.AI.BaseURL = "https://wanchuan.example/v1"
|
||||
cfg.AutoReplyConfig.AI.Model = "wanchuan-chat"
|
||||
cfg.AutoReplyConfig.AI.VisionBaseURL = ""
|
||||
cfg.AutoReplyConfig.AI.VisionModel = "wanchuan-vl" // 与聊天模型不同,属用户显式选择
|
||||
|
||||
cfg.ApplyDefaults()
|
||||
|
||||
if cfg.AutoReplyConfig.AI.VisionModel != "wanchuan-vl" {
|
||||
t.Errorf("Expected explicit different vision model preserved, got %q", cfg.AutoReplyConfig.AI.VisionModel)
|
||||
}
|
||||
}
|
||||
|
||||
// DashScope 网关:视觉模型为空或文本模型时仍应回退到专用视觉模型,不受上面改动影响。
|
||||
func TestApplyDefaultsDashScopeStillFallsBackToVisionModel(t *testing.T) {
|
||||
cfg := NewDefaultConfig()
|
||||
cfg.AutoReplyConfig.AI.BaseURL = "https://dashscope.aliyuncs.com/compatible-mode/v1"
|
||||
cfg.AutoReplyConfig.AI.Model = "qwen-turbo"
|
||||
cfg.AutoReplyConfig.AI.VisionBaseURL = ""
|
||||
cfg.AutoReplyConfig.AI.VisionModel = ""
|
||||
|
||||
cfg.ApplyDefaults()
|
||||
|
||||
if cfg.AutoReplyConfig.AI.VisionModel != defaultVisionModel {
|
||||
t.Errorf("Expected DashScope vision fallback to %q, got %q", defaultVisionModel, cfg.AutoReplyConfig.AI.VisionModel)
|
||||
}
|
||||
}
|
||||
|
||||
Reference in New Issue
Block a user