mirror of
https://github.com/router-for-me/Cli-Proxy-API-Management-Center.git
synced 2026-02-02 19:00:49 +08:00
173 lines
6.5 KiB
TypeScript
173 lines
6.5 KiB
TypeScript
import { Fragment, useMemo } from 'react';
|
|
import { useTranslation } from 'react-i18next';
|
|
import { Button } from '@/components/ui/Button';
|
|
import { Card } from '@/components/ui/Card';
|
|
import { ToggleSwitch } from '@/components/ui/ToggleSwitch';
|
|
import iconGemini from '@/assets/icons/gemini.svg';
|
|
import type { GeminiKeyConfig } from '@/types';
|
|
import { maskApiKey } from '@/utils/format';
|
|
import {
|
|
buildCandidateUsageSourceIds,
|
|
calculateStatusBarData,
|
|
type KeyStats,
|
|
type UsageDetail,
|
|
} from '@/utils/usage';
|
|
import styles from '@/pages/AiProvidersPage.module.scss';
|
|
import { ProviderList } from '../ProviderList';
|
|
import { ProviderStatusBar } from '../ProviderStatusBar';
|
|
import { getStatsBySource, hasDisableAllModelsRule } from '../utils';
|
|
|
|
interface GeminiSectionProps {
|
|
configs: GeminiKeyConfig[];
|
|
keyStats: KeyStats;
|
|
usageDetails: UsageDetail[];
|
|
loading: boolean;
|
|
disableControls: boolean;
|
|
isSwitching: boolean;
|
|
onAdd: () => void;
|
|
onEdit: (index: number) => void;
|
|
onDelete: (index: number) => void;
|
|
onToggle: (index: number, enabled: boolean) => void;
|
|
}
|
|
|
|
export function GeminiSection({
|
|
configs,
|
|
keyStats,
|
|
usageDetails,
|
|
loading,
|
|
disableControls,
|
|
isSwitching,
|
|
onAdd,
|
|
onEdit,
|
|
onDelete,
|
|
onToggle,
|
|
}: GeminiSectionProps) {
|
|
const { t } = useTranslation();
|
|
const actionsDisabled = disableControls || loading || isSwitching;
|
|
const toggleDisabled = disableControls || loading || isSwitching;
|
|
|
|
const statusBarCache = useMemo(() => {
|
|
const cache = new Map<string, ReturnType<typeof calculateStatusBarData>>();
|
|
|
|
configs.forEach((config) => {
|
|
if (!config.apiKey) return;
|
|
const candidates = buildCandidateUsageSourceIds({
|
|
apiKey: config.apiKey,
|
|
prefix: config.prefix,
|
|
});
|
|
if (!candidates.length) return;
|
|
const candidateSet = new Set(candidates);
|
|
const filteredDetails = usageDetails.filter((detail) => candidateSet.has(detail.source));
|
|
cache.set(config.apiKey, calculateStatusBarData(filteredDetails));
|
|
});
|
|
|
|
return cache;
|
|
}, [configs, usageDetails]);
|
|
|
|
return (
|
|
<>
|
|
<Card
|
|
title={
|
|
<span className={styles.cardTitle}>
|
|
<img src={iconGemini} alt="" className={styles.cardTitleIcon} />
|
|
{t('ai_providers.gemini_title')}
|
|
</span>
|
|
}
|
|
extra={
|
|
<Button size="sm" onClick={onAdd} disabled={actionsDisabled}>
|
|
{t('ai_providers.gemini_add_button')}
|
|
</Button>
|
|
}
|
|
>
|
|
<ProviderList<GeminiKeyConfig>
|
|
items={configs}
|
|
loading={loading}
|
|
keyField={(item) => item.apiKey}
|
|
emptyTitle={t('ai_providers.gemini_empty_title')}
|
|
emptyDescription={t('ai_providers.gemini_empty_desc')}
|
|
onEdit={onEdit}
|
|
onDelete={onDelete}
|
|
actionsDisabled={actionsDisabled}
|
|
getRowDisabled={(item) => hasDisableAllModelsRule(item.excludedModels)}
|
|
renderExtraActions={(item, index) => (
|
|
<ToggleSwitch
|
|
label={t('ai_providers.config_toggle_label')}
|
|
checked={!hasDisableAllModelsRule(item.excludedModels)}
|
|
disabled={toggleDisabled}
|
|
onChange={(value) => void onToggle(index, value)}
|
|
/>
|
|
)}
|
|
renderContent={(item, index) => {
|
|
const stats = getStatsBySource(item.apiKey, keyStats, item.prefix);
|
|
const headerEntries = Object.entries(item.headers || {});
|
|
const configDisabled = hasDisableAllModelsRule(item.excludedModels);
|
|
const excludedModels = item.excludedModels ?? [];
|
|
const statusData = statusBarCache.get(item.apiKey) || calculateStatusBarData([]);
|
|
|
|
return (
|
|
<Fragment>
|
|
<div className="item-title">
|
|
{t('ai_providers.gemini_item_title')} #{index + 1}
|
|
</div>
|
|
<div className={styles.fieldRow}>
|
|
<span className={styles.fieldLabel}>{t('common.api_key')}:</span>
|
|
<span className={styles.fieldValue}>{maskApiKey(item.apiKey)}</span>
|
|
</div>
|
|
{item.prefix && (
|
|
<div className={styles.fieldRow}>
|
|
<span className={styles.fieldLabel}>{t('common.prefix')}:</span>
|
|
<span className={styles.fieldValue}>{item.prefix}</span>
|
|
</div>
|
|
)}
|
|
{item.baseUrl && (
|
|
<div className={styles.fieldRow}>
|
|
<span className={styles.fieldLabel}>{t('common.base_url')}:</span>
|
|
<span className={styles.fieldValue}>{item.baseUrl}</span>
|
|
</div>
|
|
)}
|
|
{headerEntries.length > 0 && (
|
|
<div className={styles.headerBadgeList}>
|
|
{headerEntries.map(([key, value]) => (
|
|
<span key={key} className={styles.headerBadge}>
|
|
<strong>{key}:</strong> {value}
|
|
</span>
|
|
))}
|
|
</div>
|
|
)}
|
|
{configDisabled && (
|
|
<div className="status-badge warning" style={{ marginTop: 8, marginBottom: 0 }}>
|
|
{t('ai_providers.config_disabled_badge')}
|
|
</div>
|
|
)}
|
|
{excludedModels.length ? (
|
|
<div className={styles.excludedModelsSection}>
|
|
<div className={styles.excludedModelsLabel}>
|
|
{t('ai_providers.excluded_models_count', { count: excludedModels.length })}
|
|
</div>
|
|
<div className={styles.modelTagList}>
|
|
{excludedModels.map((model) => (
|
|
<span key={model} className={`${styles.modelTag} ${styles.excludedModelTag}`}>
|
|
<span className={styles.modelName}>{model}</span>
|
|
</span>
|
|
))}
|
|
</div>
|
|
</div>
|
|
) : null}
|
|
<div className={styles.cardStats}>
|
|
<span className={`${styles.statPill} ${styles.statSuccess}`}>
|
|
{t('stats.success')}: {stats.success}
|
|
</span>
|
|
<span className={`${styles.statPill} ${styles.statFailure}`}>
|
|
{t('stats.failure')}: {stats.failure}
|
|
</span>
|
|
</div>
|
|
<ProviderStatusBar statusData={statusData} />
|
|
</Fragment>
|
|
);
|
|
}}
|
|
/>
|
|
</Card>
|
|
</>
|
|
);
|
|
}
|