CodeReview/frontend/src/components/system/SystemConfig.tsx

570 lines
26 KiB
TypeScript
Raw Normal View History

/**
* System Config Component
* Cyberpunk Terminal Aesthetic
*/
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 {
Settings, Save, RotateCcw, Eye, EyeOff, CheckCircle2, AlertCircle,
Info, Zap, Globe, PlayCircle, Brain
} from "lucide-react";
import { toast } from "sonner";
import { api } from "@/shared/api/database";
import EmbeddingConfig from "@/components/agent/EmbeddingConfig";
// LLM Providers - 2025
const LLM_PROVIDERS = [
{ 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 等' },
];
const DEFAULT_MODELS: Record<string, string> = {
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',
};
interface SystemConfigData {
llmProvider: string; llmApiKey: string; llmModel: string; llmBaseUrl: string;
llmTimeout: number; llmTemperature: number; llmMaxTokens: number;
2025-12-16 16:36:08 +08:00
githubToken: string; gitlabToken: string; giteaToken: string;
maxAnalyzeFiles: number; llmConcurrency: number; llmGapMs: number; outputLanguage: string;
}
export function SystemConfig() {
const [config, setConfig] = useState<SystemConfigData | null>(null);
const [loading, setLoading] = useState(true);
const [showApiKey, setShowApiKey] = useState(false);
const [hasChanges, setHasChanges] = useState(false);
const [testingLLM, setTestingLLM] = useState(false);
const [llmTestResult, setLlmTestResult] = useState<{ success: boolean; message: string } | null>(null);
useEffect(() => { loadConfig(); }, []);
const loadConfig = async () => {
try {
setLoading(true);
console.log('[SystemConfig] 开始加载配置...');
const backendConfig = await api.getUserConfig();
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 || '',
2025-12-16 16:36:08 +08:00
giteaToken: otherConfig.giteaToken || '',
maxAnalyzeFiles: otherConfig.maxAnalyzeFiles ?? 0,
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,
2025-12-16 16:36:08 +08:00
githubToken: '', gitlabToken: '', giteaToken: '',
maxAnalyzeFiles: 0, llmConcurrency: 3, llmGapMs: 2000, outputLanguage: 'zh-CN',
});
}
} catch (error) {
console.error('Failed to load config:', error);
setConfig({
llmProvider: 'openai', llmApiKey: '', llmModel: '', llmBaseUrl: '',
llmTimeout: 150000, llmTemperature: 0.1, llmMaxTokens: 4096,
2025-12-16 16:36:08 +08:00
githubToken: '', gitlabToken: '', giteaToken: '',
maxAnalyzeFiles: 0, llmConcurrency: 3, llmGapMs: 2000, outputLanguage: 'zh-CN',
});
} finally {
setLoading(false);
}
};
const saveConfig = async () => {
if (!config) return;
try {
const savedConfig = await api.updateUserConfig({
llmConfig: {
llmProvider: config.llmProvider, llmApiKey: config.llmApiKey,
llmModel: config.llmModel, llmBaseUrl: config.llmBaseUrl,
llmTimeout: config.llmTimeout, llmTemperature: config.llmTemperature,
llmMaxTokens: config.llmMaxTokens,
},
otherConfig: {
2025-12-16 16:36:08 +08:00
githubToken: config.githubToken, gitlabToken: config.gitlabToken, giteaToken: config.giteaToken,
maxAnalyzeFiles: config.maxAnalyzeFiles, llmConcurrency: config.llmConcurrency,
llmGapMs: config.llmGapMs, outputLanguage: config.outputLanguage,
},
});
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 || '',
2025-12-16 16:36:08 +08:00
giteaToken: otherConfig.giteaToken || '',
maxAnalyzeFiles: otherConfig.maxAnalyzeFiles ?? 0,
llmConcurrency: otherConfig.llmConcurrency || 3,
llmGapMs: otherConfig.llmGapMs || 2000,
outputLanguage: otherConfig.outputLanguage || 'zh-CN',
});
}
setHasChanges(false);
toast.success("配置已保存!");
} catch (error) {
toast.error(`保存失败: ${error instanceof Error ? error.message : '未知错误'}`);
}
};
const resetConfig = async () => {
if (!window.confirm("确定要重置为默认配置吗?")) return;
try {
await api.deleteUserConfig();
await loadConfig();
setHasChanges(false);
toast.success("已重置为默认配置");
} catch (error) {
toast.error(`重置失败: ${error instanceof Error ? error.message : '未知错误'}`);
}
};
const updateConfig = (key: keyof SystemConfigData, value: string | number) => {
if (!config) return;
setConfig(prev => prev ? { ...prev, [key]: value } : null);
setHasChanges(true);
};
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);
}
};
if (loading || !config) {
return (
<div className="flex items-center justify-center min-h-[400px]">
<div className="text-center space-y-4">
<div className="loading-spinner mx-auto" />
<p className="text-gray-500 font-mono text-sm uppercase tracking-wider">...</p>
</div>
</div>
);
}
const currentProvider = LLM_PROVIDERS.find(p => p.value === config.llmProvider);
const isConfigured = config.llmApiKey !== '' || config.llmProvider === 'ollama';
return (
<div className="space-y-6">
{/* Status Bar */}
<div className={`cyber-card p-4 ${isConfigured ? 'border-emerald-500/30' : 'border-amber-500/30'}`}>
<div className="flex items-center justify-between flex-wrap gap-4">
<div className="flex items-center gap-4">
<Info className="h-5 w-5 text-sky-400" />
<span className="font-mono text-sm">
{isConfigured ? (
<span className="text-emerald-400 flex items-center gap-2">
<CheckCircle2 className="h-4 w-4" /> LLM ({currentProvider?.label})
</span>
) : (
<span className="text-amber-400 flex items-center gap-2">
<AlertCircle className="h-4 w-4" /> LLM API Key
</span>
)}
</span>
</div>
<div className="flex gap-2">
{hasChanges && (
<Button onClick={saveConfig} size="sm" className="cyber-btn-primary h-8">
<Save className="w-3 h-3 mr-2" />
</Button>
)}
<Button onClick={resetConfig} variant="outline" size="sm" className="cyber-btn-ghost h-8">
<RotateCcw className="w-3 h-3 mr-2" />
</Button>
</div>
</div>
</div>
<Tabs defaultValue="llm" className="w-full">
<TabsList className="grid w-full grid-cols-4 bg-gray-900/50 border border-gray-800 p-1 h-auto gap-1 rounded-lg mb-6">
<TabsTrigger value="llm" className="data-[state=active]:bg-primary data-[state=active]:text-white font-mono font-bold uppercase py-2.5 text-gray-400 transition-all rounded text-xs flex items-center gap-2">
<Zap className="w-3 h-3" /> LLM
</TabsTrigger>
<TabsTrigger value="embedding" className="data-[state=active]:bg-primary data-[state=active]:text-white font-mono font-bold uppercase py-2.5 text-gray-400 transition-all rounded text-xs flex items-center gap-2">
<Brain className="w-3 h-3" />
</TabsTrigger>
<TabsTrigger value="analysis" className="data-[state=active]:bg-primary data-[state=active]:text-white font-mono font-bold uppercase py-2.5 text-gray-400 transition-all rounded text-xs flex items-center gap-2">
<Settings className="w-3 h-3" />
</TabsTrigger>
<TabsTrigger value="git" className="data-[state=active]:bg-primary data-[state=active]:text-white font-mono font-bold uppercase py-2.5 text-gray-400 transition-all rounded text-xs flex items-center gap-2">
<Globe className="w-3 h-3" /> Git
</TabsTrigger>
</TabsList>
{/* LLM Config */}
<TabsContent value="llm" className="space-y-6">
<div className="cyber-card p-6 space-y-6">
{/* Provider Selection */}
<div className="space-y-2">
<Label className="text-xs font-bold text-gray-500 uppercase"> LLM </Label>
<Select value={config.llmProvider} onValueChange={(v) => updateConfig('llmProvider', v)}>
<SelectTrigger className="h-12 cyber-input">
<SelectValue />
</SelectTrigger>
<SelectContent className="bg-[#0c0c12] border-gray-700">
<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-500">- {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-500">- {p.hint}</span>
</span>
</SelectItem>
))}
</SelectContent>
</Select>
</div>
{/* API Key */}
{config.llmProvider !== 'ollama' && (
<div className="space-y-2">
<Label className="text-xs font-bold text-gray-500 uppercase">API Key</Label>
<div className="flex gap-2">
<Input
type={showApiKey ? 'text' : 'password'}
value={config.llmApiKey}
onChange={(e) => updateConfig('llmApiKey', e.target.value)}
placeholder={config.llmProvider === 'baidu' ? 'API_KEY:SECRET_KEY 格式' : '输入你的 API Key'}
className="h-12 cyber-input"
/>
<Button
variant="outline"
size="icon"
onClick={() => setShowApiKey(!showApiKey)}
className="h-12 w-12 cyber-btn-ghost"
>
{showApiKey ? <EyeOff className="h-4 w-4" /> : <Eye className="h-4 w-4" />}
</Button>
</div>
</div>
)}
{/* Model and Base URL */}
<div className="grid grid-cols-1 md:grid-cols-2 gap-4">
<div className="space-y-2">
<Label className="text-xs font-bold text-gray-500 uppercase"> ()</Label>
<Input
value={config.llmModel}
onChange={(e) => updateConfig('llmModel', e.target.value)}
placeholder={`默认: ${DEFAULT_MODELS[config.llmProvider] || 'auto'}`}
className="h-10 cyber-input"
/>
</div>
<div className="space-y-2">
<Label className="text-xs font-bold text-gray-500 uppercase">API Base URL ()</Label>
<Input
value={config.llmBaseUrl}
onChange={(e) => updateConfig('llmBaseUrl', e.target.value)}
placeholder="留空使用官方地址,或填入中转站地址"
className="h-10 cyber-input"
/>
</div>
</div>
{/* Test Connection */}
<div className="pt-4 border-t border-gray-800 border-dashed flex items-center justify-between flex-wrap gap-4">
<div className="text-sm">
<span className="font-bold text-gray-300"></span>
<span className="text-gray-500 ml-2"></span>
</div>
<Button
onClick={testLLMConnection}
disabled={testingLLM || (!isConfigured && config.llmProvider !== 'ollama')}
className="cyber-btn-primary h-10"
>
{testingLLM ? (
<>
<div className="loading-spinner w-4 h-4 mr-2" />
...
</>
) : (
<>
<PlayCircle className="w-4 h-4 mr-2" />
</>
)}
</Button>
</div>
{llmTestResult && (
<div className={`p-3 rounded-lg ${llmTestResult.success ? 'bg-emerald-500/10 border border-emerald-500/30' : 'bg-rose-500/10 border border-rose-500/30'}`}>
<div className="flex items-center gap-2 text-sm">
{llmTestResult.success ? (
<CheckCircle2 className="h-4 w-4 text-emerald-400" />
) : (
<AlertCircle className="h-4 w-4 text-rose-400" />
)}
<span className={llmTestResult.success ? 'text-emerald-300/80' : 'text-rose-300/80'}>
{llmTestResult.message}
</span>
</div>
</div>
)}
{/* Advanced Parameters */}
<details className="pt-4 border-t border-gray-800 border-dashed">
<summary className="font-bold uppercase cursor-pointer hover:text-primary text-gray-400 text-sm"></summary>
<div className="grid grid-cols-1 md:grid-cols-3 gap-4 mt-4">
<div className="space-y-2">
<Label className="text-xs text-gray-500 uppercase"> ()</Label>
<Input
type="number"
value={config.llmTimeout}
onChange={(e) => updateConfig('llmTimeout', Number(e.target.value))}
className="h-10 cyber-input"
/>
</div>
<div className="space-y-2">
<Label className="text-xs text-gray-500 uppercase"> (0-2)</Label>
<Input
type="number"
step="0.1"
min="0"
max="2"
value={config.llmTemperature}
onChange={(e) => updateConfig('llmTemperature', Number(e.target.value))}
className="h-10 cyber-input"
/>
</div>
<div className="space-y-2">
<Label className="text-xs text-gray-500 uppercase"> Tokens</Label>
<Input
type="number"
value={config.llmMaxTokens}
onChange={(e) => updateConfig('llmMaxTokens', Number(e.target.value))}
className="h-10 cyber-input"
/>
</div>
</div>
</details>
</div>
{/* Usage Notes */}
<div className="bg-gray-900/50 border border-gray-800 p-4 rounded-lg text-xs space-y-2">
<p className="font-bold uppercase text-gray-400 flex items-center gap-2">
<Info className="w-4 h-4 text-sky-400" />
</p>
<p className="text-gray-500"> <strong className="text-gray-400">LiteLLM </strong>: LiteLLM </p>
<p className="text-gray-500"> <strong className="text-gray-400"></strong>: MiniMax API 使</p>
<p className="text-gray-500"> <strong className="text-gray-400">API </strong>: Base URL API Key Key</p>
</div>
</TabsContent>
{/* Embedding Config */}
<TabsContent value="embedding" className="space-y-6">
<EmbeddingConfig />
</TabsContent>
{/* Analysis Parameters */}
<TabsContent value="analysis" className="space-y-6">
<div className="cyber-card p-6 space-y-6">
<div className="grid grid-cols-1 md:grid-cols-2 gap-6">
<div className="space-y-2">
<Label className="text-xs font-bold text-gray-500 uppercase"></Label>
<Input
type="number"
value={config.maxAnalyzeFiles}
onChange={(e) => updateConfig('maxAnalyzeFiles', Number(e.target.value))}
className="h-10 cyber-input"
/>
<p className="text-xs text-gray-600"></p>
</div>
<div className="space-y-2">
<Label className="text-xs font-bold text-gray-500 uppercase">LLM </Label>
<Input
type="number"
value={config.llmConcurrency}
onChange={(e) => updateConfig('llmConcurrency', Number(e.target.value))}
className="h-10 cyber-input"
/>
<p className="text-xs text-gray-600"> LLM </p>
</div>
<div className="space-y-2">
<Label className="text-xs font-bold text-gray-500 uppercase"> ()</Label>
<Input
type="number"
value={config.llmGapMs}
onChange={(e) => updateConfig('llmGapMs', Number(e.target.value))}
className="h-10 cyber-input"
/>
<p className="text-xs text-gray-600"></p>
</div>
<div className="space-y-2">
<Label className="text-xs font-bold text-gray-500 uppercase"></Label>
<Select value={config.outputLanguage} onValueChange={(v) => updateConfig('outputLanguage', v)}>
<SelectTrigger className="h-10 cyber-input">
<SelectValue />
</SelectTrigger>
<SelectContent className="bg-[#0c0c12] border-gray-700">
<SelectItem value="zh-CN" className="font-mono">🇨🇳 </SelectItem>
<SelectItem value="en-US" className="font-mono">🇺🇸 English</SelectItem>
</SelectContent>
</Select>
<p className="text-xs text-gray-600"></p>
</div>
</div>
</div>
</TabsContent>
{/* Git Integration */}
<TabsContent value="git" className="space-y-6">
<div className="cyber-card p-6 space-y-6">
<div className="space-y-2">
<Label className="text-xs font-bold text-gray-500 uppercase">GitHub Token ()</Label>
<Input
type="password"
value={config.githubToken}
onChange={(e) => updateConfig('githubToken', e.target.value)}
placeholder="ghp_xxxxxxxxxxxx"
className="h-10 cyber-input"
/>
<p className="text-xs text-gray-600">
访:{' '}
<a href="https://github.com/settings/tokens" target="_blank" rel="noopener noreferrer" className="text-primary hover:underline">
github.com/settings/tokens
</a>
</p>
</div>
<div className="space-y-2">
<Label className="text-xs font-bold text-gray-500 uppercase">GitLab Token ()</Label>
<Input
type="password"
value={config.gitlabToken}
onChange={(e) => updateConfig('gitlabToken', e.target.value)}
placeholder="glpat-xxxxxxxxxxxx"
className="h-10 cyber-input"
/>
<p className="text-xs text-gray-600">
访:{' '}
<a href="https://gitlab.com/-/profile/personal_access_tokens" target="_blank" rel="noopener noreferrer" className="text-primary hover:underline">
gitlab.com/-/profile/personal_access_tokens
</a>
</p>
</div>
2025-12-16 16:36:08 +08:00
<div className="space-y-2">
<Label className="text-xs font-bold text-gray-500 uppercase">Gitea Token ()</Label>
<Input
type="password"
value={config.giteaToken}
onChange={(e) => updateConfig('giteaToken', e.target.value)}
placeholder="sha1_xxxxxxxxxxxx"
className="h-10 cyber-input"
/>
<p className="text-xs text-gray-600">
访 Gitea :{' '}
<span className="text-primary">
[your-gitea-instance]/user/settings/applications
</span>
</p>
</div>
<div className="bg-gray-900/50 border border-gray-800 p-4 rounded-lg text-xs">
<p className="font-bold text-gray-400 flex items-center gap-2 mb-2">
<Info className="w-4 h-4 text-sky-400" />
</p>
<p className="text-gray-500"> Token</p>
<p className="text-gray-500"> Token</p>
</div>
</div>
</TabsContent>
</Tabs>
{/* Floating Save Button */}
{hasChanges && (
<div className="fixed bottom-6 right-6 cyber-card p-4 z-50">
<Button onClick={saveConfig} className="cyber-btn-primary h-12">
<Save className="w-4 h-4 mr-2" />
</Button>
</div>
)}
</div>
);
}