2025-10-26 13:38:17 +08:00
|
|
|
|
import { useState, useEffect } from "react";
|
|
|
|
|
|
import { Button } from "@/components/ui/button";
|
|
|
|
|
|
import { Input } from "@/components/ui/input";
|
|
|
|
|
|
import { Label } from "@/components/ui/label";
|
|
|
|
|
|
import { Select, SelectContent, SelectItem, SelectTrigger, SelectValue } from "@/components/ui/select";
|
|
|
|
|
|
import { Tabs, TabsContent, TabsList, TabsTrigger } from "@/components/ui/tabs";
|
|
|
|
|
|
import {
|
2025-11-28 16:41:39 +08:00
|
|
|
|
Settings, Save, RotateCcw, Eye, EyeOff, CheckCircle2, AlertCircle,
|
2025-12-11 19:09:10 +08:00
|
|
|
|
Info, Zap, Globe, PlayCircle, Loader2, Brain
|
2025-10-26 13:38:17 +08:00
|
|
|
|
} from "lucide-react";
|
|
|
|
|
|
import { toast } from "sonner";
|
2025-11-26 21:11:12 +08:00
|
|
|
|
import { api } from "@/shared/api/database";
|
2025-12-11 19:09:10 +08:00
|
|
|
|
import EmbeddingConfig from "@/components/agent/EmbeddingConfig";
|
2025-10-26 13:38:17 +08:00
|
|
|
|
|
feat(llm): enhance LLM connection testing with improved error handling and adapter instantiation
- Bypass LLMFactory cache during connection tests to ensure fresh API calls with latest configuration
- Directly instantiate native adapters (Baidu, Minimax, Doubao) and LiteLLM adapter based on provider type
- Add comprehensive error handling in LiteLLM adapter with specific exception catching for authentication, rate limiting, and connection errors
- Implement user-friendly error messages for common failure scenarios (invalid API key, authentication failure, timeout, connection issues)
- Add response validation to detect and report empty API responses
- Disable LiteLLM internal caching to guarantee actual API calls during testing
- Update available models list with 2025 latest models across all providers (Gemini, OpenAI, Claude, Qwen, DeepSeek, etc.)
- Improve error message clarity and debugging information in config endpoint
2025-11-28 16:53:01 +08:00
|
|
|
|
// LLM 提供商配置 - 2025年最新
|
2025-10-26 13:38:17 +08:00
|
|
|
|
const LLM_PROVIDERS = [
|
feat(llm): enhance LLM connection testing with improved error handling and adapter instantiation
- Bypass LLMFactory cache during connection tests to ensure fresh API calls with latest configuration
- Directly instantiate native adapters (Baidu, Minimax, Doubao) and LiteLLM adapter based on provider type
- Add comprehensive error handling in LiteLLM adapter with specific exception catching for authentication, rate limiting, and connection errors
- Implement user-friendly error messages for common failure scenarios (invalid API key, authentication failure, timeout, connection issues)
- Add response validation to detect and report empty API responses
- Disable LiteLLM internal caching to guarantee actual API calls during testing
- Update available models list with 2025 latest models across all providers (Gemini, OpenAI, Claude, Qwen, DeepSeek, etc.)
- Improve error message clarity and debugging information in config endpoint
2025-11-28 16:53:01 +08:00
|
|
|
|
{ value: 'openai', label: 'OpenAI GPT', icon: '🟢', category: 'litellm', hint: 'gpt-5, gpt-5-mini, o3 等' },
|
|
|
|
|
|
{ value: 'claude', label: 'Anthropic Claude', icon: '🟣', category: 'litellm', hint: 'claude-sonnet-4.5, claude-opus-4 等' },
|
|
|
|
|
|
{ value: 'gemini', label: 'Google Gemini', icon: '🔵', category: 'litellm', hint: 'gemini-3-pro, gemini-3-flash 等' },
|
|
|
|
|
|
{ value: 'deepseek', label: 'DeepSeek', icon: '🔷', category: 'litellm', hint: 'deepseek-v3.1-terminus, deepseek-v3 等' },
|
|
|
|
|
|
{ value: 'qwen', label: '通义千问', icon: '🟠', category: 'litellm', hint: 'qwen3-max-instruct, qwen3-plus 等' },
|
|
|
|
|
|
{ value: 'zhipu', label: '智谱AI (GLM)', icon: '🔴', category: 'litellm', hint: 'glm-4.6, glm-4.5-flash 等' },
|
|
|
|
|
|
{ value: 'moonshot', label: 'Moonshot (Kimi)', icon: '🌙', category: 'litellm', hint: 'kimi-k2, kimi-k1.5 等' },
|
|
|
|
|
|
{ value: 'ollama', label: 'Ollama 本地', icon: '🖥️', category: 'litellm', hint: 'llama3.3-70b, qwen3-8b 等' },
|
|
|
|
|
|
{ value: 'baidu', label: '百度文心', icon: '📘', category: 'native', hint: 'ernie-4.5 (需要 API_KEY:SECRET_KEY)' },
|
|
|
|
|
|
{ value: 'minimax', label: 'MiniMax', icon: '⚡', category: 'native', hint: 'minimax-m2, minimax-m1 等' },
|
|
|
|
|
|
{ value: 'doubao', label: '字节豆包', icon: '🎯', category: 'native', hint: 'doubao-1.6-pro, doubao-1.5-pro 等' },
|
2025-10-26 13:38:17 +08:00
|
|
|
|
];
|
|
|
|
|
|
|
2025-11-28 16:41:39 +08:00
|
|
|
|
const DEFAULT_MODELS: Record<string, string> = {
|
feat(llm): enhance LLM connection testing with improved error handling and adapter instantiation
- Bypass LLMFactory cache during connection tests to ensure fresh API calls with latest configuration
- Directly instantiate native adapters (Baidu, Minimax, Doubao) and LiteLLM adapter based on provider type
- Add comprehensive error handling in LiteLLM adapter with specific exception catching for authentication, rate limiting, and connection errors
- Implement user-friendly error messages for common failure scenarios (invalid API key, authentication failure, timeout, connection issues)
- Add response validation to detect and report empty API responses
- Disable LiteLLM internal caching to guarantee actual API calls during testing
- Update available models list with 2025 latest models across all providers (Gemini, OpenAI, Claude, Qwen, DeepSeek, etc.)
- Improve error message clarity and debugging information in config endpoint
2025-11-28 16:53:01 +08:00
|
|
|
|
openai: 'gpt-5', claude: 'claude-sonnet-4.5', gemini: 'gemini-3-pro',
|
|
|
|
|
|
deepseek: 'deepseek-v3.1-terminus', qwen: 'qwen3-max-instruct', zhipu: 'glm-4.6', moonshot: 'kimi-k2',
|
|
|
|
|
|
ollama: 'llama3.3-70b', baidu: 'ernie-4.5', minimax: 'minimax-m2', doubao: 'doubao-1.6-pro',
|
2025-10-26 13:38:17 +08:00
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
interface SystemConfigData {
|
2025-11-28 16:41:39 +08:00
|
|
|
|
llmProvider: string; llmApiKey: string; llmModel: string; llmBaseUrl: string;
|
|
|
|
|
|
llmTimeout: number; llmTemperature: number; llmMaxTokens: number;
|
|
|
|
|
|
githubToken: string; gitlabToken: string;
|
|
|
|
|
|
maxAnalyzeFiles: number; llmConcurrency: number; llmGapMs: number; outputLanguage: string;
|
2025-10-26 13:38:17 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
export function SystemConfig() {
|
2025-11-26 21:11:12 +08:00
|
|
|
|
const [config, setConfig] = useState<SystemConfigData | null>(null);
|
|
|
|
|
|
const [loading, setLoading] = useState(true);
|
2025-11-28 16:41:39 +08:00
|
|
|
|
const [showApiKey, setShowApiKey] = useState(false);
|
2025-10-26 13:38:17 +08:00
|
|
|
|
const [hasChanges, setHasChanges] = useState(false);
|
2025-11-28 16:41:39 +08:00
|
|
|
|
const [testingLLM, setTestingLLM] = useState(false);
|
|
|
|
|
|
const [llmTestResult, setLlmTestResult] = useState<{ success: boolean; message: string } | null>(null);
|
2025-10-26 13:38:17 +08:00
|
|
|
|
|
2025-11-28 16:41:39 +08:00
|
|
|
|
useEffect(() => { loadConfig(); }, []);
|
2025-10-26 13:38:17 +08:00
|
|
|
|
|
2025-11-26 21:11:12 +08:00
|
|
|
|
const loadConfig = async () => {
|
2025-10-26 13:38:17 +08:00
|
|
|
|
try {
|
2025-11-26 21:11:12 +08:00
|
|
|
|
setLoading(true);
|
2025-11-28 17:51:17 +08:00
|
|
|
|
console.log('[SystemConfig] 开始加载配置...');
|
|
|
|
|
|
|
|
|
|
|
|
// 后端 /config/me 已经返回合并后的配置(用户配置优先,然后是系统默认配置)
|
2025-11-26 21:11:12 +08:00
|
|
|
|
const backendConfig = await api.getUserConfig();
|
2025-11-28 16:41:39 +08:00
|
|
|
|
|
2025-11-28 17:51:17 +08:00
|
|
|
|
console.log('[SystemConfig] 后端返回的原始数据:', JSON.stringify(backendConfig, null, 2));
|
|
|
|
|
|
|
|
|
|
|
|
if (backendConfig) {
|
|
|
|
|
|
// 直接使用后端返回的合并配置
|
|
|
|
|
|
const llmConfig = backendConfig.llmConfig || {};
|
|
|
|
|
|
const otherConfig = backendConfig.otherConfig || {};
|
|
|
|
|
|
|
|
|
|
|
|
const newConfig = {
|
|
|
|
|
|
llmProvider: llmConfig.llmProvider || 'openai',
|
|
|
|
|
|
llmApiKey: llmConfig.llmApiKey || '',
|
|
|
|
|
|
llmModel: llmConfig.llmModel || '',
|
|
|
|
|
|
llmBaseUrl: llmConfig.llmBaseUrl || '',
|
|
|
|
|
|
llmTimeout: llmConfig.llmTimeout || 150000,
|
|
|
|
|
|
llmTemperature: llmConfig.llmTemperature ?? 0.1,
|
|
|
|
|
|
llmMaxTokens: llmConfig.llmMaxTokens || 4096,
|
|
|
|
|
|
githubToken: otherConfig.githubToken || '',
|
|
|
|
|
|
gitlabToken: otherConfig.gitlabToken || '',
|
|
|
|
|
|
maxAnalyzeFiles: otherConfig.maxAnalyzeFiles || 50,
|
|
|
|
|
|
llmConcurrency: otherConfig.llmConcurrency || 3,
|
|
|
|
|
|
llmGapMs: otherConfig.llmGapMs || 2000,
|
|
|
|
|
|
outputLanguage: otherConfig.outputLanguage || 'zh-CN',
|
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
console.log('[SystemConfig] 解析后的配置:', newConfig);
|
|
|
|
|
|
setConfig(newConfig);
|
|
|
|
|
|
|
|
|
|
|
|
console.log('✓ 配置已加载:', {
|
|
|
|
|
|
provider: llmConfig.llmProvider,
|
|
|
|
|
|
hasApiKey: !!llmConfig.llmApiKey,
|
|
|
|
|
|
model: llmConfig.llmModel,
|
|
|
|
|
|
});
|
|
|
|
|
|
} else {
|
|
|
|
|
|
console.warn('[SystemConfig] 后端返回空数据,使用默认配置');
|
|
|
|
|
|
// 如果获取失败,使用默认值
|
|
|
|
|
|
setConfig({
|
|
|
|
|
|
llmProvider: 'openai', llmApiKey: '', llmModel: '', llmBaseUrl: '',
|
|
|
|
|
|
llmTimeout: 150000, llmTemperature: 0.1, llmMaxTokens: 4096,
|
|
|
|
|
|
githubToken: '', gitlabToken: '',
|
|
|
|
|
|
maxAnalyzeFiles: 50, llmConcurrency: 3, llmGapMs: 2000, outputLanguage: 'zh-CN',
|
|
|
|
|
|
});
|
|
|
|
|
|
}
|
2025-10-26 13:38:17 +08:00
|
|
|
|
} catch (error) {
|
|
|
|
|
|
console.error('Failed to load config:', error);
|
2025-11-28 16:41:39 +08:00
|
|
|
|
setConfig({
|
|
|
|
|
|
llmProvider: 'openai', llmApiKey: '', llmModel: '', llmBaseUrl: '',
|
|
|
|
|
|
llmTimeout: 150000, llmTemperature: 0.1, llmMaxTokens: 4096,
|
|
|
|
|
|
githubToken: '', gitlabToken: '',
|
|
|
|
|
|
maxAnalyzeFiles: 50, llmConcurrency: 3, llmGapMs: 2000, outputLanguage: 'zh-CN',
|
|
|
|
|
|
});
|
2025-11-26 21:11:12 +08:00
|
|
|
|
} finally {
|
|
|
|
|
|
setLoading(false);
|
2025-10-26 13:38:17 +08:00
|
|
|
|
}
|
|
|
|
|
|
};
|
|
|
|
|
|
|
2025-11-26 21:11:12 +08:00
|
|
|
|
const saveConfig = async () => {
|
2025-11-28 16:41:39 +08:00
|
|
|
|
if (!config) return;
|
2025-10-26 13:38:17 +08:00
|
|
|
|
try {
|
2025-11-28 17:51:17 +08:00
|
|
|
|
const savedConfig = await api.updateUserConfig({
|
2025-11-28 16:41:39 +08:00
|
|
|
|
llmConfig: {
|
|
|
|
|
|
llmProvider: config.llmProvider, llmApiKey: config.llmApiKey,
|
|
|
|
|
|
llmModel: config.llmModel, llmBaseUrl: config.llmBaseUrl,
|
|
|
|
|
|
llmTimeout: config.llmTimeout, llmTemperature: config.llmTemperature,
|
|
|
|
|
|
llmMaxTokens: config.llmMaxTokens,
|
|
|
|
|
|
},
|
|
|
|
|
|
otherConfig: {
|
|
|
|
|
|
githubToken: config.githubToken, gitlabToken: config.gitlabToken,
|
|
|
|
|
|
maxAnalyzeFiles: config.maxAnalyzeFiles, llmConcurrency: config.llmConcurrency,
|
|
|
|
|
|
llmGapMs: config.llmGapMs, outputLanguage: config.outputLanguage,
|
|
|
|
|
|
},
|
2025-11-26 21:11:12 +08:00
|
|
|
|
});
|
2025-11-28 17:51:17 +08:00
|
|
|
|
|
|
|
|
|
|
// 使用后端返回的数据更新本地状态,确保数据同步
|
|
|
|
|
|
if (savedConfig) {
|
|
|
|
|
|
const llmConfig = savedConfig.llmConfig || {};
|
|
|
|
|
|
const otherConfig = savedConfig.otherConfig || {};
|
|
|
|
|
|
setConfig({
|
|
|
|
|
|
llmProvider: llmConfig.llmProvider || config.llmProvider,
|
|
|
|
|
|
llmApiKey: llmConfig.llmApiKey || '',
|
|
|
|
|
|
llmModel: llmConfig.llmModel || '',
|
|
|
|
|
|
llmBaseUrl: llmConfig.llmBaseUrl || '',
|
|
|
|
|
|
llmTimeout: llmConfig.llmTimeout || 150000,
|
|
|
|
|
|
llmTemperature: llmConfig.llmTemperature ?? 0.1,
|
|
|
|
|
|
llmMaxTokens: llmConfig.llmMaxTokens || 4096,
|
|
|
|
|
|
githubToken: otherConfig.githubToken || '',
|
|
|
|
|
|
gitlabToken: otherConfig.gitlabToken || '',
|
|
|
|
|
|
maxAnalyzeFiles: otherConfig.maxAnalyzeFiles || 50,
|
|
|
|
|
|
llmConcurrency: otherConfig.llmConcurrency || 3,
|
|
|
|
|
|
llmGapMs: otherConfig.llmGapMs || 2000,
|
|
|
|
|
|
outputLanguage: otherConfig.outputLanguage || 'zh-CN',
|
|
|
|
|
|
});
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2025-10-26 13:38:17 +08:00
|
|
|
|
setHasChanges(false);
|
2025-11-28 16:41:39 +08:00
|
|
|
|
toast.success("配置已保存!");
|
2025-10-26 13:38:17 +08:00
|
|
|
|
} catch (error) {
|
2025-11-28 16:41:39 +08:00
|
|
|
|
toast.error(`保存失败: ${error instanceof Error ? error.message : '未知错误'}`);
|
2025-10-26 13:38:17 +08:00
|
|
|
|
}
|
|
|
|
|
|
};
|
|
|
|
|
|
|
2025-11-26 21:11:12 +08:00
|
|
|
|
const resetConfig = async () => {
|
2025-11-28 16:41:39 +08:00
|
|
|
|
if (!window.confirm("确定要重置为默认配置吗?")) return;
|
|
|
|
|
|
try {
|
|
|
|
|
|
await api.deleteUserConfig();
|
|
|
|
|
|
await loadConfig();
|
|
|
|
|
|
setHasChanges(false);
|
|
|
|
|
|
toast.success("已重置为默认配置");
|
|
|
|
|
|
} catch (error) {
|
|
|
|
|
|
toast.error(`重置失败: ${error instanceof Error ? error.message : '未知错误'}`);
|
2025-10-26 13:38:17 +08:00
|
|
|
|
}
|
|
|
|
|
|
};
|
|
|
|
|
|
|
2025-11-28 16:41:39 +08:00
|
|
|
|
const updateConfig = (key: keyof SystemConfigData, value: string | number) => {
|
2025-11-26 21:11:12 +08:00
|
|
|
|
if (!config) return;
|
|
|
|
|
|
setConfig(prev => prev ? { ...prev, [key]: value } : null);
|
2025-10-26 13:38:17 +08:00
|
|
|
|
setHasChanges(true);
|
|
|
|
|
|
};
|
|
|
|
|
|
|
2025-11-28 16:41:39 +08:00
|
|
|
|
const testLLMConnection = async () => {
|
|
|
|
|
|
if (!config) return;
|
|
|
|
|
|
if (!config.llmApiKey && config.llmProvider !== 'ollama') {
|
|
|
|
|
|
toast.error('请先配置 API Key');
|
|
|
|
|
|
return;
|
|
|
|
|
|
}
|
|
|
|
|
|
setTestingLLM(true);
|
|
|
|
|
|
setLlmTestResult(null);
|
|
|
|
|
|
try {
|
|
|
|
|
|
const result = await api.testLLMConnection({
|
|
|
|
|
|
provider: config.llmProvider,
|
|
|
|
|
|
apiKey: config.llmApiKey,
|
|
|
|
|
|
model: config.llmModel || undefined,
|
|
|
|
|
|
baseUrl: config.llmBaseUrl || undefined,
|
|
|
|
|
|
});
|
|
|
|
|
|
setLlmTestResult(result);
|
|
|
|
|
|
if (result.success) toast.success(`连接成功!模型: ${result.model}`);
|
|
|
|
|
|
else toast.error(`连接失败: ${result.message}`);
|
|
|
|
|
|
} catch (error) {
|
|
|
|
|
|
const msg = error instanceof Error ? error.message : '未知错误';
|
|
|
|
|
|
setLlmTestResult({ success: false, message: msg });
|
|
|
|
|
|
toast.error(`测试失败: ${msg}`);
|
|
|
|
|
|
} finally {
|
|
|
|
|
|
setTestingLLM(false);
|
|
|
|
|
|
}
|
2025-10-26 13:38:17 +08:00
|
|
|
|
};
|
|
|
|
|
|
|
2025-11-26 21:11:12 +08:00
|
|
|
|
if (loading || !config) {
|
|
|
|
|
|
return (
|
|
|
|
|
|
<div className="flex items-center justify-center min-h-[400px]">
|
|
|
|
|
|
<div className="text-center">
|
2025-11-27 21:33:51 +08:00
|
|
|
|
<div className="animate-spin rounded-none h-12 w-12 border-4 border-black border-t-transparent mx-auto mb-4"></div>
|
2025-11-28 16:41:39 +08:00
|
|
|
|
<p className="text-black font-mono font-bold uppercase">加载配置中...</p>
|
2025-11-26 21:11:12 +08:00
|
|
|
|
</div>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
);
|
|
|
|
|
|
}
|
2025-10-26 13:38:17 +08:00
|
|
|
|
|
2025-11-28 16:41:39 +08:00
|
|
|
|
const currentProvider = LLM_PROVIDERS.find(p => p.value === config.llmProvider);
|
|
|
|
|
|
const isConfigured = config.llmApiKey !== '' || config.llmProvider === 'ollama';
|
|
|
|
|
|
|
2025-10-26 13:38:17 +08:00
|
|
|
|
return (
|
|
|
|
|
|
<div className="space-y-6">
|
2025-11-28 16:41:39 +08:00
|
|
|
|
{/* 状态栏 */}
|
|
|
|
|
|
<div className="bg-blue-50 border-2 border-blue-500 p-4 flex items-center justify-between shadow-[4px_4px_0px_0px_rgba(59,130,246,1)]">
|
|
|
|
|
|
<div className="flex items-center gap-4 font-mono text-sm">
|
|
|
|
|
|
<Info className="h-5 w-5 text-blue-600" />
|
|
|
|
|
|
<span className="font-bold">
|
|
|
|
|
|
{isConfigured ? (
|
|
|
|
|
|
<span className="text-green-600 flex items-center gap-1">
|
|
|
|
|
|
<CheckCircle2 className="h-4 w-4" /> LLM 已配置 ({currentProvider?.label})
|
|
|
|
|
|
</span>
|
2025-10-26 13:38:17 +08:00
|
|
|
|
) : (
|
2025-11-28 16:41:39 +08:00
|
|
|
|
<span className="text-orange-600 flex items-center gap-1">
|
|
|
|
|
|
<AlertCircle className="h-4 w-4" /> 请配置 LLM API Key
|
|
|
|
|
|
</span>
|
2025-10-26 13:38:17 +08:00
|
|
|
|
)}
|
2025-11-28 16:41:39 +08:00
|
|
|
|
</span>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
<div className="flex gap-2">
|
|
|
|
|
|
{hasChanges && (
|
|
|
|
|
|
<Button onClick={saveConfig} size="sm" className="retro-btn bg-black text-white border-2 border-black hover:bg-gray-800 rounded-none h-8 font-bold uppercase">
|
|
|
|
|
|
<Save className="w-3 h-3 mr-2" /> 保存
|
|
|
|
|
|
</Button>
|
|
|
|
|
|
)}
|
|
|
|
|
|
<Button onClick={resetConfig} variant="outline" size="sm" className="retro-btn bg-white text-black border-2 border-black hover:bg-gray-100 rounded-none h-8 font-bold uppercase">
|
|
|
|
|
|
<RotateCcw className="w-3 h-3 mr-2" /> 重置
|
|
|
|
|
|
</Button>
|
2025-11-27 21:33:51 +08:00
|
|
|
|
</div>
|
|
|
|
|
|
</div>
|
2025-10-26 13:38:17 +08:00
|
|
|
|
|
|
|
|
|
|
<Tabs defaultValue="llm" className="w-full">
|
2025-12-11 19:09:10 +08:00
|
|
|
|
<TabsList className="grid w-full grid-cols-4 bg-transparent border-2 border-black p-0 h-auto gap-0 mb-6">
|
2025-11-27 21:33:51 +08:00
|
|
|
|
<TabsTrigger value="llm" className="rounded-none border-r-2 border-black data-[state=active]:bg-black data-[state=active]:text-white font-mono font-bold uppercase h-10 text-xs">
|
2025-11-28 16:41:39 +08:00
|
|
|
|
<Zap className="w-3 h-3 mr-2" /> LLM 配置
|
2025-10-26 13:38:17 +08:00
|
|
|
|
</TabsTrigger>
|
2025-12-11 19:09:10 +08:00
|
|
|
|
<TabsTrigger value="embedding" className="rounded-none border-r-2 border-black data-[state=active]:bg-black data-[state=active]:text-white font-mono font-bold uppercase h-10 text-xs">
|
|
|
|
|
|
<Brain className="w-3 h-3 mr-2" /> 嵌入模型
|
|
|
|
|
|
</TabsTrigger>
|
2025-11-27 21:33:51 +08:00
|
|
|
|
<TabsTrigger value="analysis" className="rounded-none border-r-2 border-black data-[state=active]:bg-black data-[state=active]:text-white font-mono font-bold uppercase h-10 text-xs">
|
2025-11-28 16:41:39 +08:00
|
|
|
|
<Settings className="w-3 h-3 mr-2" /> 分析参数
|
2025-10-26 13:38:17 +08:00
|
|
|
|
</TabsTrigger>
|
2025-11-28 16:41:39 +08:00
|
|
|
|
<TabsTrigger value="git" className="rounded-none data-[state=active]:bg-black data-[state=active]:text-white font-mono font-bold uppercase h-10 text-xs">
|
|
|
|
|
|
<Globe className="w-3 h-3 mr-2" /> Git 集成
|
2025-10-26 13:38:17 +08:00
|
|
|
|
</TabsTrigger>
|
|
|
|
|
|
</TabsList>
|
|
|
|
|
|
|
2025-11-28 16:41:39 +08:00
|
|
|
|
{/* LLM 配置 - 简化版 */}
|
2025-10-26 13:38:17 +08:00
|
|
|
|
<TabsContent value="llm" className="space-y-6">
|
2025-11-28 16:41:39 +08:00
|
|
|
|
<div className="retro-card bg-white border-2 border-black shadow-[4px_4px_0px_0px_rgba(0,0,0,1)] p-6 space-y-6">
|
|
|
|
|
|
{/* 提供商选择 */}
|
|
|
|
|
|
<div className="space-y-2">
|
|
|
|
|
|
<Label className="font-mono font-bold uppercase">选择 LLM 提供商</Label>
|
|
|
|
|
|
<Select value={config.llmProvider} onValueChange={(v) => updateConfig('llmProvider', v)}>
|
|
|
|
|
|
<SelectTrigger className="h-12 bg-gray-50 border-2 border-black rounded-none font-mono">
|
|
|
|
|
|
<SelectValue />
|
|
|
|
|
|
</SelectTrigger>
|
|
|
|
|
|
<SelectContent className="border-2 border-black rounded-none">
|
|
|
|
|
|
<div className="px-2 py-1.5 text-xs font-bold text-gray-500 uppercase">LiteLLM 统一适配 (推荐)</div>
|
|
|
|
|
|
{LLM_PROVIDERS.filter(p => p.category === 'litellm').map(p => (
|
|
|
|
|
|
<SelectItem key={p.value} value={p.value} className="font-mono">
|
|
|
|
|
|
<span className="flex items-center gap-2">
|
|
|
|
|
|
<span>{p.icon}</span>
|
|
|
|
|
|
<span>{p.label}</span>
|
|
|
|
|
|
<span className="text-xs text-gray-400">- {p.hint}</span>
|
|
|
|
|
|
</span>
|
|
|
|
|
|
</SelectItem>
|
|
|
|
|
|
))}
|
|
|
|
|
|
<div className="px-2 py-1.5 text-xs font-bold text-gray-500 uppercase mt-2">原生适配器</div>
|
|
|
|
|
|
{LLM_PROVIDERS.filter(p => p.category === 'native').map(p => (
|
|
|
|
|
|
<SelectItem key={p.value} value={p.value} className="font-mono">
|
|
|
|
|
|
<span className="flex items-center gap-2">
|
|
|
|
|
|
<span>{p.icon}</span>
|
|
|
|
|
|
<span>{p.label}</span>
|
|
|
|
|
|
<span className="text-xs text-gray-400">- {p.hint}</span>
|
|
|
|
|
|
</span>
|
|
|
|
|
|
</SelectItem>
|
|
|
|
|
|
))}
|
|
|
|
|
|
</SelectContent>
|
|
|
|
|
|
</Select>
|
2025-11-27 21:33:51 +08:00
|
|
|
|
</div>
|
2025-10-26 13:38:17 +08:00
|
|
|
|
|
2025-11-28 16:41:39 +08:00
|
|
|
|
{/* API Key */}
|
|
|
|
|
|
{config.llmProvider !== 'ollama' && (
|
2025-10-26 13:38:17 +08:00
|
|
|
|
<div className="space-y-2">
|
2025-11-28 16:41:39 +08:00
|
|
|
|
<Label className="font-mono font-bold uppercase">API Key</Label>
|
2025-10-26 13:38:17 +08:00
|
|
|
|
<div className="flex gap-2">
|
|
|
|
|
|
<Input
|
2025-11-28 16:41:39 +08:00
|
|
|
|
type={showApiKey ? 'text' : 'password'}
|
2025-10-26 13:38:17 +08:00
|
|
|
|
value={config.llmApiKey}
|
|
|
|
|
|
onChange={(e) => updateConfig('llmApiKey', e.target.value)}
|
2025-11-28 16:41:39 +08:00
|
|
|
|
placeholder={config.llmProvider === 'baidu' ? 'API_KEY:SECRET_KEY 格式' : '输入你的 API Key'}
|
|
|
|
|
|
className="h-12 bg-gray-50 border-2 border-black rounded-none font-mono"
|
2025-10-26 13:38:17 +08:00
|
|
|
|
/>
|
2025-11-28 16:41:39 +08:00
|
|
|
|
<Button variant="outline" size="icon" onClick={() => setShowApiKey(!showApiKey)}
|
|
|
|
|
|
className="h-12 w-12 border-2 border-black rounded-none">
|
|
|
|
|
|
{showApiKey ? <EyeOff className="h-4 w-4" /> : <Eye className="h-4 w-4" />}
|
2025-10-26 13:38:17 +08:00
|
|
|
|
</Button>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
</div>
|
2025-11-28 16:41:39 +08:00
|
|
|
|
)}
|
2025-10-26 13:38:17 +08:00
|
|
|
|
|
2025-11-28 16:41:39 +08:00
|
|
|
|
{/* 模型和 Base URL */}
|
|
|
|
|
|
<div className="grid grid-cols-1 md:grid-cols-2 gap-4">
|
2025-10-26 13:38:17 +08:00
|
|
|
|
<div className="space-y-2">
|
2025-11-28 16:41:39 +08:00
|
|
|
|
<Label className="font-mono font-bold uppercase">模型名称 (可选)</Label>
|
2025-10-26 13:38:17 +08:00
|
|
|
|
<Input
|
|
|
|
|
|
value={config.llmModel}
|
|
|
|
|
|
onChange={(e) => updateConfig('llmModel', e.target.value)}
|
2025-11-28 16:41:39 +08:00
|
|
|
|
placeholder={`默认: ${DEFAULT_MODELS[config.llmProvider] || 'auto'}`}
|
|
|
|
|
|
className="h-10 bg-gray-50 border-2 border-black rounded-none font-mono"
|
2025-10-26 13:38:17 +08:00
|
|
|
|
/>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
<div className="space-y-2">
|
2025-11-28 16:41:39 +08:00
|
|
|
|
<Label className="font-mono font-bold uppercase">API Base URL (可选)</Label>
|
2025-10-26 13:38:17 +08:00
|
|
|
|
<Input
|
|
|
|
|
|
value={config.llmBaseUrl}
|
|
|
|
|
|
onChange={(e) => updateConfig('llmBaseUrl', e.target.value)}
|
2025-11-28 16:41:39 +08:00
|
|
|
|
placeholder="留空使用官方地址,或填入中转站地址"
|
|
|
|
|
|
className="h-10 bg-gray-50 border-2 border-black rounded-none font-mono"
|
2025-10-26 13:38:17 +08:00
|
|
|
|
/>
|
2025-11-28 16:41:39 +08:00
|
|
|
|
</div>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
|
|
|
|
|
|
{/* 测试连接 */}
|
|
|
|
|
|
<div className="pt-4 border-t-2 border-black border-dashed flex items-center justify-between">
|
|
|
|
|
|
<div className="text-sm font-mono">
|
|
|
|
|
|
<span className="font-bold">测试连接</span>
|
|
|
|
|
|
<span className="text-gray-500 ml-2">验证配置是否正确</span>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
<Button onClick={testLLMConnection} disabled={testingLLM || (!isConfigured && config.llmProvider !== 'ollama')}
|
|
|
|
|
|
className="retro-btn bg-black text-white border-2 border-black hover:bg-gray-800 rounded-none h-10 font-bold uppercase">
|
|
|
|
|
|
{testingLLM ? <><Loader2 className="w-4 h-4 mr-2 animate-spin" /> 测试中...</> : <><PlayCircle className="w-4 h-4 mr-2" /> 测试</>}
|
|
|
|
|
|
</Button>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
{llmTestResult && (
|
|
|
|
|
|
<div className={`p-3 border-2 ${llmTestResult.success ? 'border-green-500 bg-green-50' : 'border-red-500 bg-red-50'}`}>
|
|
|
|
|
|
<div className="flex items-center gap-2 font-mono text-sm">
|
|
|
|
|
|
{llmTestResult.success ? <CheckCircle2 className="h-4 w-4 text-green-600" /> : <AlertCircle className="h-4 w-4 text-red-600" />}
|
|
|
|
|
|
<span className={llmTestResult.success ? 'text-green-800' : 'text-red-800'}>{llmTestResult.message}</span>
|
2025-10-26 13:38:17 +08:00
|
|
|
|
</div>
|
|
|
|
|
|
</div>
|
2025-11-28 16:41:39 +08:00
|
|
|
|
)}
|
2025-10-26 13:38:17 +08:00
|
|
|
|
|
2025-11-28 16:41:39 +08:00
|
|
|
|
{/* 高级参数 - 折叠 */}
|
|
|
|
|
|
<details className="pt-4 border-t-2 border-black border-dashed">
|
|
|
|
|
|
<summary className="font-mono font-bold uppercase cursor-pointer hover:text-blue-600">高级参数</summary>
|
|
|
|
|
|
<div className="grid grid-cols-1 md:grid-cols-3 gap-4 mt-4">
|
2025-10-26 13:38:17 +08:00
|
|
|
|
<div className="space-y-2">
|
2025-11-28 16:41:39 +08:00
|
|
|
|
<Label className="font-mono text-xs uppercase">超时 (毫秒)</Label>
|
|
|
|
|
|
<Input type="number" value={config.llmTimeout} onChange={(e) => updateConfig('llmTimeout', Number(e.target.value))}
|
|
|
|
|
|
className="h-10 bg-gray-50 border-2 border-black rounded-none font-mono" />
|
2025-10-26 13:38:17 +08:00
|
|
|
|
</div>
|
|
|
|
|
|
<div className="space-y-2">
|
2025-11-28 16:41:39 +08:00
|
|
|
|
<Label className="font-mono text-xs uppercase">温度 (0-2)</Label>
|
|
|
|
|
|
<Input type="number" step="0.1" min="0" max="2" value={config.llmTemperature}
|
2025-10-26 13:38:17 +08:00
|
|
|
|
onChange={(e) => updateConfig('llmTemperature', Number(e.target.value))}
|
2025-11-28 16:41:39 +08:00
|
|
|
|
className="h-10 bg-gray-50 border-2 border-black rounded-none font-mono" />
|
2025-10-26 13:38:17 +08:00
|
|
|
|
</div>
|
|
|
|
|
|
<div className="space-y-2">
|
2025-11-28 16:41:39 +08:00
|
|
|
|
<Label className="font-mono text-xs uppercase">最大 Tokens</Label>
|
|
|
|
|
|
<Input type="number" value={config.llmMaxTokens} onChange={(e) => updateConfig('llmMaxTokens', Number(e.target.value))}
|
|
|
|
|
|
className="h-10 bg-gray-50 border-2 border-black rounded-none font-mono" />
|
2025-10-26 13:38:17 +08:00
|
|
|
|
</div>
|
|
|
|
|
|
</div>
|
2025-11-28 16:41:39 +08:00
|
|
|
|
</details>
|
2025-11-27 21:33:51 +08:00
|
|
|
|
</div>
|
2025-10-26 13:38:17 +08:00
|
|
|
|
|
2025-11-28 16:41:39 +08:00
|
|
|
|
{/* 使用说明 */}
|
|
|
|
|
|
<div className="bg-gray-50 border-2 border-black p-4 font-mono text-xs space-y-2">
|
|
|
|
|
|
<p className="font-bold uppercase">💡 配置说明</p>
|
|
|
|
|
|
<p>• <strong>LiteLLM 统一适配</strong>: 大多数提供商通过 LiteLLM 统一处理,支持自动重试和负载均衡</p>
|
|
|
|
|
|
<p>• <strong>原生适配器</strong>: 百度、MiniMax、豆包因 API 格式特殊,使用专用适配器</p>
|
|
|
|
|
|
<p>• <strong>API 中转站</strong>: 在 Base URL 填入中转站地址即可,API Key 填中转站提供的 Key</p>
|
2025-11-27 21:33:51 +08:00
|
|
|
|
</div>
|
2025-10-26 13:38:17 +08:00
|
|
|
|
</TabsContent>
|
|
|
|
|
|
|
2025-12-11 19:09:10 +08:00
|
|
|
|
{/* 嵌入模型配置 */}
|
|
|
|
|
|
<TabsContent value="embedding" className="space-y-6">
|
|
|
|
|
|
<EmbeddingConfig />
|
|
|
|
|
|
</TabsContent>
|
|
|
|
|
|
|
2025-11-28 16:41:39 +08:00
|
|
|
|
{/* 分析参数 */}
|
2025-10-26 13:38:17 +08:00
|
|
|
|
<TabsContent value="analysis" className="space-y-6">
|
2025-11-28 16:41:39 +08:00
|
|
|
|
<div className="retro-card bg-white border-2 border-black shadow-[4px_4px_0px_0px_rgba(0,0,0,1)] p-6 space-y-6">
|
|
|
|
|
|
<div className="grid grid-cols-1 md:grid-cols-2 gap-6">
|
2025-10-26 13:38:17 +08:00
|
|
|
|
<div className="space-y-2">
|
2025-11-28 16:41:39 +08:00
|
|
|
|
<Label className="font-mono font-bold uppercase">最大分析文件数</Label>
|
|
|
|
|
|
<Input type="number" value={config.maxAnalyzeFiles}
|
2025-10-26 13:38:17 +08:00
|
|
|
|
onChange={(e) => updateConfig('maxAnalyzeFiles', Number(e.target.value))}
|
2025-11-28 16:41:39 +08:00
|
|
|
|
className="h-10 bg-gray-50 border-2 border-black rounded-none font-mono" />
|
|
|
|
|
|
<p className="text-xs text-gray-500 font-mono">单次任务最多处理的文件数量</p>
|
2025-10-26 13:38:17 +08:00
|
|
|
|
</div>
|
|
|
|
|
|
<div className="space-y-2">
|
2025-11-28 16:41:39 +08:00
|
|
|
|
<Label className="font-mono font-bold uppercase">LLM 并发数</Label>
|
|
|
|
|
|
<Input type="number" value={config.llmConcurrency}
|
2025-10-26 13:38:17 +08:00
|
|
|
|
onChange={(e) => updateConfig('llmConcurrency', Number(e.target.value))}
|
2025-11-28 16:41:39 +08:00
|
|
|
|
className="h-10 bg-gray-50 border-2 border-black rounded-none font-mono" />
|
|
|
|
|
|
<p className="text-xs text-gray-500 font-mono">同时发送的 LLM 请求数量</p>
|
2025-10-26 13:38:17 +08:00
|
|
|
|
</div>
|
|
|
|
|
|
<div className="space-y-2">
|
2025-11-28 16:41:39 +08:00
|
|
|
|
<Label className="font-mono font-bold uppercase">请求间隔 (毫秒)</Label>
|
|
|
|
|
|
<Input type="number" value={config.llmGapMs}
|
2025-10-26 13:38:17 +08:00
|
|
|
|
onChange={(e) => updateConfig('llmGapMs', Number(e.target.value))}
|
2025-11-28 16:41:39 +08:00
|
|
|
|
className="h-10 bg-gray-50 border-2 border-black rounded-none font-mono" />
|
|
|
|
|
|
<p className="text-xs text-gray-500 font-mono">每个请求之间的延迟时间</p>
|
2025-10-26 13:38:17 +08:00
|
|
|
|
</div>
|
|
|
|
|
|
<div className="space-y-2">
|
2025-11-28 16:41:39 +08:00
|
|
|
|
<Label className="font-mono font-bold uppercase">输出语言</Label>
|
|
|
|
|
|
<Select value={config.outputLanguage} onValueChange={(v) => updateConfig('outputLanguage', v)}>
|
|
|
|
|
|
<SelectTrigger className="h-10 bg-gray-50 border-2 border-black rounded-none font-mono">
|
2025-10-26 13:38:17 +08:00
|
|
|
|
<SelectValue />
|
|
|
|
|
|
</SelectTrigger>
|
2025-11-28 16:41:39 +08:00
|
|
|
|
<SelectContent className="border-2 border-black rounded-none">
|
|
|
|
|
|
<SelectItem value="zh-CN" className="font-mono">🇨🇳 中文</SelectItem>
|
|
|
|
|
|
<SelectItem value="en-US" className="font-mono">🇺🇸 English</SelectItem>
|
2025-10-26 13:38:17 +08:00
|
|
|
|
</SelectContent>
|
|
|
|
|
|
</Select>
|
2025-11-28 16:41:39 +08:00
|
|
|
|
<p className="text-xs text-gray-500 font-mono">代码审查结果的输出语言</p>
|
2025-10-26 13:38:17 +08:00
|
|
|
|
</div>
|
2025-11-27 21:33:51 +08:00
|
|
|
|
</div>
|
|
|
|
|
|
</div>
|
2025-10-26 13:38:17 +08:00
|
|
|
|
</TabsContent>
|
|
|
|
|
|
|
2025-11-28 16:41:39 +08:00
|
|
|
|
{/* Git 集成 */}
|
|
|
|
|
|
<TabsContent value="git" className="space-y-6">
|
|
|
|
|
|
<div className="retro-card bg-white border-2 border-black shadow-[4px_4px_0px_0px_rgba(0,0,0,1)] p-6 space-y-6">
|
|
|
|
|
|
<div className="space-y-2">
|
|
|
|
|
|
<Label className="font-mono font-bold uppercase">GitHub Token (可选)</Label>
|
|
|
|
|
|
<Input
|
|
|
|
|
|
type="password"
|
|
|
|
|
|
value={config.githubToken}
|
|
|
|
|
|
onChange={(e) => updateConfig('githubToken', e.target.value)}
|
|
|
|
|
|
placeholder="ghp_xxxxxxxxxxxx"
|
|
|
|
|
|
className="h-10 bg-gray-50 border-2 border-black rounded-none font-mono"
|
|
|
|
|
|
/>
|
|
|
|
|
|
<p className="text-xs text-gray-500 font-mono">
|
|
|
|
|
|
用于访问私有仓库。获取: <a href="https://github.com/settings/tokens" target="_blank" rel="noopener noreferrer" className="text-blue-600 hover:underline">github.com/settings/tokens</a>
|
|
|
|
|
|
</p>
|
2025-11-27 21:33:51 +08:00
|
|
|
|
</div>
|
2025-11-28 16:41:39 +08:00
|
|
|
|
<div className="space-y-2">
|
|
|
|
|
|
<Label className="font-mono font-bold uppercase">GitLab Token (可选)</Label>
|
|
|
|
|
|
<Input
|
|
|
|
|
|
type="password"
|
|
|
|
|
|
value={config.gitlabToken}
|
|
|
|
|
|
onChange={(e) => updateConfig('gitlabToken', e.target.value)}
|
|
|
|
|
|
placeholder="glpat-xxxxxxxxxxxx"
|
|
|
|
|
|
className="h-10 bg-gray-50 border-2 border-black rounded-none font-mono"
|
|
|
|
|
|
/>
|
|
|
|
|
|
<p className="text-xs text-gray-500 font-mono">
|
|
|
|
|
|
用于访问私有仓库。获取: <a href="https://gitlab.com/-/profile/personal_access_tokens" target="_blank" rel="noopener noreferrer" className="text-blue-600 hover:underline">gitlab.com/-/profile/personal_access_tokens</a>
|
|
|
|
|
|
</p>
|
2025-11-27 21:33:51 +08:00
|
|
|
|
</div>
|
2025-11-28 16:41:39 +08:00
|
|
|
|
<div className="bg-gray-50 border-2 border-black p-4 font-mono text-xs">
|
|
|
|
|
|
<p className="font-bold">💡 提示</p>
|
|
|
|
|
|
<p>• 公开仓库无需配置 Token</p>
|
|
|
|
|
|
<p>• 私有仓库需要配置对应平台的 Token</p>
|
2025-11-27 21:33:51 +08:00
|
|
|
|
</div>
|
|
|
|
|
|
</div>
|
2025-10-26 13:38:17 +08:00
|
|
|
|
</TabsContent>
|
|
|
|
|
|
</Tabs>
|
|
|
|
|
|
|
2025-11-28 16:41:39 +08:00
|
|
|
|
{/* 底部保存按钮 */}
|
2025-10-26 13:38:17 +08:00
|
|
|
|
{hasChanges && (
|
2025-11-28 16:41:39 +08:00
|
|
|
|
<div className="fixed bottom-6 right-6 bg-white border-2 border-black shadow-[4px_4px_0px_0px_rgba(0,0,0,1)] p-4 z-50">
|
|
|
|
|
|
<Button onClick={saveConfig} className="retro-btn bg-black text-white border-2 border-black hover:bg-gray-800 rounded-none h-12 font-bold uppercase">
|
|
|
|
|
|
<Save className="w-4 h-4 mr-2" /> 保存所有更改
|
2025-10-26 13:38:17 +08:00
|
|
|
|
</Button>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
)}
|
|
|
|
|
|
</div>
|
|
|
|
|
|
);
|
|
|
|
|
|
}
|