mirror of
https://github.com/router-for-me/Cli-Proxy-API-Management-Center.git
synced 2026-02-03 03:10:50 +08:00
Compare commits
14 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
dea106cf47 | ||
|
|
76ef1b68af | ||
|
|
39a003bdd4 | ||
|
|
b1426ccefc | ||
|
|
a9df58cba7 | ||
|
|
f6563490a6 | ||
|
|
18c1ba6c3c | ||
|
|
c2627cac3e | ||
|
|
df472119e7 | ||
|
|
10f2262753 | ||
|
|
39d86d133a | ||
|
|
ddbd7d00bd | ||
|
|
e44beb541f | ||
|
|
aecd5875d6 |
35
.github/workflows/release.yml
vendored
35
.github/workflows/release.yml
vendored
@@ -15,6 +15,9 @@ jobs:
|
|||||||
steps:
|
steps:
|
||||||
- name: Checkout code
|
- name: Checkout code
|
||||||
uses: actions/checkout@v4
|
uses: actions/checkout@v4
|
||||||
|
with:
|
||||||
|
fetch-depth: 0
|
||||||
|
fetch-tags: true
|
||||||
|
|
||||||
- name: Setup Node.js
|
- name: Setup Node.js
|
||||||
uses: actions/setup-node@v4
|
uses: actions/setup-node@v4
|
||||||
@@ -36,27 +39,25 @@ jobs:
|
|||||||
mv index.html management.html
|
mv index.html management.html
|
||||||
ls -lh management.html
|
ls -lh management.html
|
||||||
|
|
||||||
|
- name: Generate release notes
|
||||||
|
run: |
|
||||||
|
set -euo pipefail
|
||||||
|
current_tag="${GITHUB_REF_NAME}"
|
||||||
|
previous_tag="$(git tag --list 'v*' --sort=-v:refname | grep -v "^${current_tag}$" | head -n 1 || true)"
|
||||||
|
if [ -n "${previous_tag}" ]; then
|
||||||
|
range="${previous_tag}..${current_tag}"
|
||||||
|
else
|
||||||
|
range="${current_tag}"
|
||||||
|
fi
|
||||||
|
|
||||||
|
: > release-notes.md
|
||||||
|
git log --pretty=format:"- %h %s" "${range}" >> release-notes.md
|
||||||
|
|
||||||
- name: Create Release
|
- name: Create Release
|
||||||
uses: softprops/action-gh-release@v1
|
uses: softprops/action-gh-release@v1
|
||||||
with:
|
with:
|
||||||
files: dist/management.html
|
files: dist/management.html
|
||||||
body: |
|
body_path: release-notes.md
|
||||||
## CLI Proxy API Management Center - ${{ github.ref_name }}
|
|
||||||
|
|
||||||
### Download and Usage
|
|
||||||
1. Download the `management.html` file
|
|
||||||
2. Open it directly in your browser
|
|
||||||
3. All assets (CSS, JavaScript, images) are bundled into this single file
|
|
||||||
|
|
||||||
### Features
|
|
||||||
- Single file, no external dependencies required
|
|
||||||
- Complete management interface for CLI Proxy API
|
|
||||||
- Support for local and remote connections
|
|
||||||
- Multi-language support (Chinese/English)
|
|
||||||
- Dark/Light theme support
|
|
||||||
|
|
||||||
---
|
|
||||||
🤖 Generated with GitHub Actions
|
|
||||||
draft: false
|
draft: false
|
||||||
prerelease: false
|
prerelease: false
|
||||||
env:
|
env:
|
||||||
|
|||||||
3
.gitignore
vendored
3
.gitignore
vendored
@@ -8,6 +8,8 @@ pnpm-debug.log*
|
|||||||
lerna-debug.log*
|
lerna-debug.log*
|
||||||
api.md
|
api.md
|
||||||
usage.json
|
usage.json
|
||||||
|
CLAUDE.md
|
||||||
|
AGENTS.md
|
||||||
|
|
||||||
node_modules
|
node_modules
|
||||||
dist
|
dist
|
||||||
@@ -15,6 +17,7 @@ dist-ssr
|
|||||||
*.local
|
*.local
|
||||||
|
|
||||||
# Editor directories and files
|
# Editor directories and files
|
||||||
|
settings.local.json
|
||||||
.vscode/*
|
.vscode/*
|
||||||
!.vscode/extensions.json
|
!.vscode/extensions.json
|
||||||
.idea
|
.idea
|
||||||
|
|||||||
21
src/App.tsx
21
src/App.tsx
@@ -1,6 +1,7 @@
|
|||||||
import { useEffect } from 'react';
|
import { useCallback, useEffect, useState } from 'react';
|
||||||
import { HashRouter, Navigate, Route, Routes } from 'react-router-dom';
|
import { HashRouter, Navigate, Route, Routes } from 'react-router-dom';
|
||||||
import { LoginPage } from '@/pages/LoginPage';
|
import { LoginPage } from '@/pages/LoginPage';
|
||||||
|
import { DashboardPage } from '@/pages/DashboardPage';
|
||||||
import { SettingsPage } from '@/pages/SettingsPage';
|
import { SettingsPage } from '@/pages/SettingsPage';
|
||||||
import { ApiKeysPage } from '@/pages/ApiKeysPage';
|
import { ApiKeysPage } from '@/pages/ApiKeysPage';
|
||||||
import { AiProvidersPage } from '@/pages/AiProvidersPage';
|
import { AiProvidersPage } from '@/pages/AiProvidersPage';
|
||||||
@@ -11,6 +12,7 @@ import { ConfigPage } from '@/pages/ConfigPage';
|
|||||||
import { LogsPage } from '@/pages/LogsPage';
|
import { LogsPage } from '@/pages/LogsPage';
|
||||||
import { SystemPage } from '@/pages/SystemPage';
|
import { SystemPage } from '@/pages/SystemPage';
|
||||||
import { NotificationContainer } from '@/components/common/NotificationContainer';
|
import { NotificationContainer } from '@/components/common/NotificationContainer';
|
||||||
|
import { SplashScreen } from '@/components/common/SplashScreen';
|
||||||
import { MainLayout } from '@/components/layout/MainLayout';
|
import { MainLayout } from '@/components/layout/MainLayout';
|
||||||
import { ProtectedRoute } from '@/router/ProtectedRoute';
|
import { ProtectedRoute } from '@/router/ProtectedRoute';
|
||||||
import { useAuthStore, useLanguageStore, useThemeStore } from '@/stores';
|
import { useAuthStore, useLanguageStore, useThemeStore } from '@/stores';
|
||||||
@@ -20,6 +22,9 @@ function App() {
|
|||||||
const language = useLanguageStore((state) => state.language);
|
const language = useLanguageStore((state) => state.language);
|
||||||
const setLanguage = useLanguageStore((state) => state.setLanguage);
|
const setLanguage = useLanguageStore((state) => state.setLanguage);
|
||||||
const restoreSession = useAuthStore((state) => state.restoreSession);
|
const restoreSession = useAuthStore((state) => state.restoreSession);
|
||||||
|
const isAuthenticated = useAuthStore((state) => state.isAuthenticated);
|
||||||
|
|
||||||
|
const [showSplash, setShowSplash] = useState(true);
|
||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
initializeTheme();
|
initializeTheme();
|
||||||
@@ -31,6 +36,15 @@ function App() {
|
|||||||
// eslint-disable-next-line react-hooks/exhaustive-deps
|
// eslint-disable-next-line react-hooks/exhaustive-deps
|
||||||
}, []); // 仅用于首屏同步 i18n 语言
|
}, []); // 仅用于首屏同步 i18n 语言
|
||||||
|
|
||||||
|
const handleSplashFinish = useCallback(() => {
|
||||||
|
setShowSplash(false);
|
||||||
|
}, []);
|
||||||
|
|
||||||
|
// 仅在已认证时显示闪屏
|
||||||
|
if (showSplash && isAuthenticated) {
|
||||||
|
return <SplashScreen onFinish={handleSplashFinish} duration={1500} />;
|
||||||
|
}
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<HashRouter>
|
<HashRouter>
|
||||||
<NotificationContainer />
|
<NotificationContainer />
|
||||||
@@ -44,7 +58,8 @@ function App() {
|
|||||||
</ProtectedRoute>
|
</ProtectedRoute>
|
||||||
}
|
}
|
||||||
>
|
>
|
||||||
<Route index element={<Navigate to="/settings" replace />} />
|
<Route index element={<DashboardPage />} />
|
||||||
|
<Route path="dashboard" element={<DashboardPage />} />
|
||||||
<Route path="settings" element={<SettingsPage />} />
|
<Route path="settings" element={<SettingsPage />} />
|
||||||
<Route path="api-keys" element={<ApiKeysPage />} />
|
<Route path="api-keys" element={<ApiKeysPage />} />
|
||||||
<Route path="ai-providers" element={<AiProvidersPage />} />
|
<Route path="ai-providers" element={<AiProvidersPage />} />
|
||||||
@@ -54,7 +69,7 @@ function App() {
|
|||||||
<Route path="config" element={<ConfigPage />} />
|
<Route path="config" element={<ConfigPage />} />
|
||||||
<Route path="logs" element={<LogsPage />} />
|
<Route path="logs" element={<LogsPage />} />
|
||||||
<Route path="system" element={<SystemPage />} />
|
<Route path="system" element={<SystemPage />} />
|
||||||
<Route path="*" element={<Navigate to="/settings" replace />} />
|
<Route path="*" element={<Navigate to="/" replace />} />
|
||||||
</Route>
|
</Route>
|
||||||
</Routes>
|
</Routes>
|
||||||
</HashRouter>
|
</HashRouter>
|
||||||
|
|||||||
106
src/components/common/SplashScreen.scss
Normal file
106
src/components/common/SplashScreen.scss
Normal file
@@ -0,0 +1,106 @@
|
|||||||
|
@use 'sass:color';
|
||||||
|
@use '../../styles/variables.scss' as *;
|
||||||
|
|
||||||
|
.splash-screen {
|
||||||
|
position: fixed;
|
||||||
|
inset: 0;
|
||||||
|
z-index: 9999;
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
justify-content: center;
|
||||||
|
background: var(--bg-primary);
|
||||||
|
opacity: 1;
|
||||||
|
transition: opacity 0.4s ease-out;
|
||||||
|
|
||||||
|
&.fade-out {
|
||||||
|
opacity: 0;
|
||||||
|
pointer-events: none;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
.splash-content {
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
align-items: center;
|
||||||
|
gap: $spacing-md;
|
||||||
|
animation: splash-enter 0.6s ease-out;
|
||||||
|
}
|
||||||
|
|
||||||
|
@keyframes splash-enter {
|
||||||
|
from {
|
||||||
|
opacity: 0;
|
||||||
|
transform: scale(0.9) translateY(20px);
|
||||||
|
}
|
||||||
|
to {
|
||||||
|
opacity: 1;
|
||||||
|
transform: scale(1) translateY(0);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
.splash-logo {
|
||||||
|
height: 80px;
|
||||||
|
width: auto;
|
||||||
|
border-radius: $radius-lg;
|
||||||
|
box-shadow: $shadow-lg;
|
||||||
|
animation: splash-logo-pulse 1.5s ease-in-out infinite;
|
||||||
|
}
|
||||||
|
|
||||||
|
@keyframes splash-logo-pulse {
|
||||||
|
0%, 100% {
|
||||||
|
transform: scale(1);
|
||||||
|
}
|
||||||
|
50% {
|
||||||
|
transform: scale(1.05);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
.splash-title {
|
||||||
|
font-size: 28px;
|
||||||
|
font-weight: 800;
|
||||||
|
color: var(--text-primary);
|
||||||
|
margin: 0;
|
||||||
|
letter-spacing: -0.5px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.splash-subtitle {
|
||||||
|
font-size: 16px;
|
||||||
|
font-weight: 500;
|
||||||
|
color: var(--text-secondary);
|
||||||
|
margin: 0;
|
||||||
|
margin-top: -8px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.splash-loader {
|
||||||
|
width: 120px;
|
||||||
|
height: 3px;
|
||||||
|
background: var(--border-color);
|
||||||
|
border-radius: $radius-full;
|
||||||
|
overflow: hidden;
|
||||||
|
margin-top: $spacing-md;
|
||||||
|
}
|
||||||
|
|
||||||
|
.splash-loader-bar {
|
||||||
|
width: 100%;
|
||||||
|
height: 100%;
|
||||||
|
background: var(--primary-color);
|
||||||
|
border-radius: $radius-full;
|
||||||
|
animation: splash-loading 1.2s ease-in-out infinite;
|
||||||
|
transform-origin: left;
|
||||||
|
}
|
||||||
|
|
||||||
|
@keyframes splash-loading {
|
||||||
|
0% {
|
||||||
|
transform: scaleX(0);
|
||||||
|
}
|
||||||
|
50% {
|
||||||
|
transform: scaleX(1);
|
||||||
|
transform-origin: left;
|
||||||
|
}
|
||||||
|
50.01% {
|
||||||
|
transform-origin: right;
|
||||||
|
}
|
||||||
|
100% {
|
||||||
|
transform: scaleX(0);
|
||||||
|
transform-origin: right;
|
||||||
|
}
|
||||||
|
}
|
||||||
40
src/components/common/SplashScreen.tsx
Normal file
40
src/components/common/SplashScreen.tsx
Normal file
@@ -0,0 +1,40 @@
|
|||||||
|
import { useEffect, useState } from 'react';
|
||||||
|
import { INLINE_LOGO_JPEG } from '@/assets/logoInline';
|
||||||
|
import './SplashScreen.scss';
|
||||||
|
|
||||||
|
interface SplashScreenProps {
|
||||||
|
onFinish: () => void;
|
||||||
|
duration?: number;
|
||||||
|
}
|
||||||
|
|
||||||
|
export function SplashScreen({ onFinish, duration = 1500 }: SplashScreenProps) {
|
||||||
|
const [fadeOut, setFadeOut] = useState(false);
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
const fadeTimer = setTimeout(() => {
|
||||||
|
setFadeOut(true);
|
||||||
|
}, duration - 400);
|
||||||
|
|
||||||
|
const finishTimer = setTimeout(() => {
|
||||||
|
onFinish();
|
||||||
|
}, duration);
|
||||||
|
|
||||||
|
return () => {
|
||||||
|
clearTimeout(fadeTimer);
|
||||||
|
clearTimeout(finishTimer);
|
||||||
|
};
|
||||||
|
}, [duration, onFinish]);
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div className={`splash-screen ${fadeOut ? 'fade-out' : ''}`}>
|
||||||
|
<div className="splash-content">
|
||||||
|
<img src={INLINE_LOGO_JPEG} alt="CPAMC" className="splash-logo" />
|
||||||
|
<h1 className="splash-title">CLI Proxy API</h1>
|
||||||
|
<p className="splash-subtitle">Management Center</p>
|
||||||
|
<div className="splash-loader">
|
||||||
|
<div className="splash-loader-bar" />
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
}
|
||||||
@@ -8,6 +8,7 @@ import {
|
|||||||
IconFileText,
|
IconFileText,
|
||||||
IconInfo,
|
IconInfo,
|
||||||
IconKey,
|
IconKey,
|
||||||
|
IconLayoutDashboard,
|
||||||
IconScrollText,
|
IconScrollText,
|
||||||
IconSettings,
|
IconSettings,
|
||||||
IconShield,
|
IconShield,
|
||||||
@@ -18,6 +19,7 @@ import { useAuthStore, useConfigStore, useLanguageStore, useNotificationStore, u
|
|||||||
import { versionApi } from '@/services/api';
|
import { versionApi } from '@/services/api';
|
||||||
|
|
||||||
const sidebarIcons: Record<string, ReactNode> = {
|
const sidebarIcons: Record<string, ReactNode> = {
|
||||||
|
dashboard: <IconLayoutDashboard size={18} />,
|
||||||
settings: <IconSlidersHorizontal size={18} />,
|
settings: <IconSlidersHorizontal size={18} />,
|
||||||
apiKeys: <IconKey size={18} />,
|
apiKeys: <IconKey size={18} />,
|
||||||
aiProviders: <IconBot size={18} />,
|
aiProviders: <IconBot size={18} />,
|
||||||
@@ -230,6 +232,7 @@ export function MainLayout() {
|
|||||||
: 'muted';
|
: 'muted';
|
||||||
|
|
||||||
const navItems = [
|
const navItems = [
|
||||||
|
{ path: '/', label: t('nav.dashboard'), icon: sidebarIcons.dashboard },
|
||||||
{ path: '/settings', label: t('nav.basic_settings'), icon: sidebarIcons.settings },
|
{ path: '/settings', label: t('nav.basic_settings'), icon: sidebarIcons.settings },
|
||||||
{ path: '/api-keys', label: t('nav.api_keys'), icon: sidebarIcons.apiKeys },
|
{ path: '/api-keys', label: t('nav.api_keys'), icon: sidebarIcons.apiKeys },
|
||||||
{ path: '/ai-providers', label: t('nav.ai_providers'), icon: sidebarIcons.aiProviders },
|
{ path: '/ai-providers', label: t('nav.ai_providers'), icon: sidebarIcons.aiProviders },
|
||||||
|
|||||||
@@ -1,7 +1,13 @@
|
|||||||
export function LoadingSpinner({ size = 20 }: { size?: number }) {
|
export function LoadingSpinner({
|
||||||
|
size = 20,
|
||||||
|
className = ''
|
||||||
|
}: {
|
||||||
|
size?: number;
|
||||||
|
className?: string;
|
||||||
|
}) {
|
||||||
return (
|
return (
|
||||||
<div
|
<div
|
||||||
className="loading-spinner"
|
className={`loading-spinner${className ? ` ${className}` : ''}`}
|
||||||
style={{ width: size, height: size, borderWidth: size / 7 }}
|
style={{ width: size, height: size, borderWidth: size / 7 }}
|
||||||
role="status"
|
role="status"
|
||||||
aria-live="polite"
|
aria-live="polite"
|
||||||
|
|||||||
@@ -303,3 +303,14 @@ export function IconCode({ size = 20, ...props }: IconProps) {
|
|||||||
</svg>
|
</svg>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
export function IconLayoutDashboard({ size = 20, ...props }: IconProps) {
|
||||||
|
return (
|
||||||
|
<svg {...baseSvgProps} width={size} height={size} {...props}>
|
||||||
|
<rect width="7" height="9" x="3" y="3" rx="1" />
|
||||||
|
<rect width="7" height="5" x="14" y="3" rx="1" />
|
||||||
|
<rect width="7" height="9" x="14" y="12" rx="1" />
|
||||||
|
<rect width="7" height="5" x="3" y="16" rx="1" />
|
||||||
|
</svg>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|||||||
@@ -81,6 +81,7 @@
|
|||||||
"status": "Connection Status:"
|
"status": "Connection Status:"
|
||||||
},
|
},
|
||||||
"nav": {
|
"nav": {
|
||||||
|
"dashboard": "Dashboard",
|
||||||
"basic_settings": "Basic Settings",
|
"basic_settings": "Basic Settings",
|
||||||
"api_keys": "API Keys",
|
"api_keys": "API Keys",
|
||||||
"ai_providers": "AI Providers",
|
"ai_providers": "AI Providers",
|
||||||
@@ -91,6 +92,27 @@
|
|||||||
"logs": "Logs Viewer",
|
"logs": "Logs Viewer",
|
||||||
"system_info": "Management Center Info"
|
"system_info": "Management Center Info"
|
||||||
},
|
},
|
||||||
|
"dashboard": {
|
||||||
|
"title": "Dashboard",
|
||||||
|
"subtitle": "Welcome to CLI Proxy API Management Center",
|
||||||
|
"openai_providers": "OpenAI Providers",
|
||||||
|
"quick_actions": "Quick Actions",
|
||||||
|
"current_config": "Current Configuration",
|
||||||
|
"management_keys": "Management Keys",
|
||||||
|
"provider_keys_detail": "G:{{gemini}} C:{{codex}} Cl:{{claude}} O:{{openai}}",
|
||||||
|
"oauth_credentials": "OAuth Credentials",
|
||||||
|
"usage_overview": "Usage Overview",
|
||||||
|
"total_requests": "Total Requests",
|
||||||
|
"total_tokens": "Total Tokens",
|
||||||
|
"rpm_30min": "RPM (30min)",
|
||||||
|
"tpm_30min": "TPM (30min)",
|
||||||
|
"models_used": "Models Used",
|
||||||
|
"no_usage_data": "No usage data available",
|
||||||
|
"view_detailed_usage": "View Detailed Stats",
|
||||||
|
"edit_settings": "Edit Settings",
|
||||||
|
"available_models": "Available Models",
|
||||||
|
"available_models_desc": "Total models from all providers"
|
||||||
|
},
|
||||||
"basic_settings": {
|
"basic_settings": {
|
||||||
"title": "Basic Settings",
|
"title": "Basic Settings",
|
||||||
"debug_title": "Debug Mode",
|
"debug_title": "Debug Mode",
|
||||||
@@ -200,8 +222,6 @@
|
|||||||
"ampcode_upstream_api_key_current": "Current Amp official key: {{key}}",
|
"ampcode_upstream_api_key_current": "Current Amp official key: {{key}}",
|
||||||
"ampcode_clear_upstream_api_key": "Clear official key",
|
"ampcode_clear_upstream_api_key": "Clear official key",
|
||||||
"ampcode_clear_upstream_api_key_confirm": "Are you sure you want to clear the Ampcode upstream API key (Amp official)?",
|
"ampcode_clear_upstream_api_key_confirm": "Are you sure you want to clear the Ampcode upstream API key (Amp official)?",
|
||||||
"ampcode_restrict_management_label": "Restrict Amp management routes to localhost",
|
|
||||||
"ampcode_restrict_management_hint": "When enabled, Amp management routes (/api/auth, /api/user, /api/threads, etc.) only accept 127.0.0.1/::1 (recommended).",
|
|
||||||
"ampcode_force_model_mappings_label": "Force model mappings",
|
"ampcode_force_model_mappings_label": "Force model mappings",
|
||||||
"ampcode_force_model_mappings_hint": "When enabled, mappings override local API-key availability checks.",
|
"ampcode_force_model_mappings_hint": "When enabled, mappings override local API-key availability checks.",
|
||||||
"ampcode_model_mappings_label": "Model mappings (from → to)",
|
"ampcode_model_mappings_label": "Model mappings (from → to)",
|
||||||
@@ -424,9 +444,10 @@
|
|||||||
"gemini_cli_oauth_title": "Gemini CLI OAuth",
|
"gemini_cli_oauth_title": "Gemini CLI OAuth",
|
||||||
"gemini_cli_oauth_button": "Start Gemini CLI Login",
|
"gemini_cli_oauth_button": "Start Gemini CLI Login",
|
||||||
"gemini_cli_oauth_hint": "Login to Google Gemini CLI service through OAuth flow, automatically obtain and save authentication files.",
|
"gemini_cli_oauth_hint": "Login to Google Gemini CLI service through OAuth flow, automatically obtain and save authentication files.",
|
||||||
"gemini_cli_project_id_label": "Google Cloud Project ID (Optional):",
|
"gemini_cli_project_id_label": "Google Cloud Project ID:",
|
||||||
"gemini_cli_project_id_placeholder": "Enter Google Cloud Project ID (optional)",
|
"gemini_cli_project_id_placeholder": "Enter Google Cloud Project ID",
|
||||||
"gemini_cli_project_id_hint": "If a project ID is specified, authentication information for that project will be used.",
|
"gemini_cli_project_id_hint": "Project ID is required for Gemini CLI OAuth.",
|
||||||
|
"gemini_cli_project_id_required": "Please enter a Google Cloud project ID.",
|
||||||
"gemini_cli_oauth_url_label": "Authorization URL:",
|
"gemini_cli_oauth_url_label": "Authorization URL:",
|
||||||
"gemini_cli_open_link": "Open Link",
|
"gemini_cli_open_link": "Open Link",
|
||||||
"gemini_cli_copy_link": "Copy Link",
|
"gemini_cli_copy_link": "Copy Link",
|
||||||
@@ -446,6 +467,16 @@
|
|||||||
"qwen_oauth_status_error": "Authentication failed:",
|
"qwen_oauth_status_error": "Authentication failed:",
|
||||||
"qwen_oauth_start_error": "Failed to start Qwen OAuth:",
|
"qwen_oauth_start_error": "Failed to start Qwen OAuth:",
|
||||||
"qwen_oauth_polling_error": "Failed to check authentication status:",
|
"qwen_oauth_polling_error": "Failed to check authentication status:",
|
||||||
|
"oauth_callback_label": "Callback URL",
|
||||||
|
"oauth_callback_placeholder": "http://localhost:1455/auth/callback?code=...&state=...",
|
||||||
|
"oauth_callback_hint": "Remote browser mode: after the provider redirects to http://localhost:..., copy the full URL and submit it here.",
|
||||||
|
"oauth_callback_button": "Submit Callback URL",
|
||||||
|
"oauth_callback_required": "Please paste the full redirect URL first.",
|
||||||
|
"oauth_callback_success": "Callback URL submitted. Continue waiting for authentication.",
|
||||||
|
"oauth_callback_error": "Failed to submit callback URL:",
|
||||||
|
"oauth_callback_upgrade_hint": "Please update CLI Proxy API or check the connection.",
|
||||||
|
"oauth_callback_status_success": "Callback URL submitted, waiting for authentication...",
|
||||||
|
"oauth_callback_status_error": "Callback URL submission failed:",
|
||||||
"missing_state": "Unable to retrieve authentication state parameter",
|
"missing_state": "Unable to retrieve authentication state parameter",
|
||||||
"iflow_oauth_title": "iFlow OAuth",
|
"iflow_oauth_title": "iFlow OAuth",
|
||||||
"iflow_oauth_button": "Start iFlow Login",
|
"iflow_oauth_button": "Start iFlow Login",
|
||||||
@@ -571,11 +602,15 @@
|
|||||||
"auto_refresh_disabled": "Auto refresh disabled",
|
"auto_refresh_disabled": "Auto refresh disabled",
|
||||||
"load_more_hint": "Scroll up to load more",
|
"load_more_hint": "Scroll up to load more",
|
||||||
"hidden_lines": "Hidden: {{count}} lines",
|
"hidden_lines": "Hidden: {{count}} lines",
|
||||||
|
"hide_management_logs": "Hide {{prefix}} logs",
|
||||||
"search_placeholder": "Search logs by content or keyword",
|
"search_placeholder": "Search logs by content or keyword",
|
||||||
"search_empty_title": "No matching logs found",
|
"search_empty_title": "No matching logs found",
|
||||||
"search_empty_desc": "Try a different keyword or clear the search filter.",
|
"search_empty_desc": "Try a different keyword or clear the filters.",
|
||||||
|
"double_click_copy_hint": "Double-click to copy raw log line",
|
||||||
|
"copy_success": "Log copied to clipboard",
|
||||||
|
"copy_failed": "Copy failed",
|
||||||
"lines": "lines",
|
"lines": "lines",
|
||||||
"removed": "Removed",
|
"removed": "Filtered",
|
||||||
"upgrade_required_title": "Please Upgrade CLI Proxy API",
|
"upgrade_required_title": "Please Upgrade CLI Proxy API",
|
||||||
"upgrade_required_desc": "The current server version does not support the logs viewing feature. Please upgrade to the latest version of CLI Proxy API to use this feature."
|
"upgrade_required_desc": "The current server version does not support the logs viewing feature. Please upgrade to the latest version of CLI Proxy API to use this feature."
|
||||||
},
|
},
|
||||||
|
|||||||
@@ -81,6 +81,7 @@
|
|||||||
"status": "连接状态:"
|
"status": "连接状态:"
|
||||||
},
|
},
|
||||||
"nav": {
|
"nav": {
|
||||||
|
"dashboard": "仪表盘",
|
||||||
"basic_settings": "基础设置",
|
"basic_settings": "基础设置",
|
||||||
"api_keys": "API 密钥",
|
"api_keys": "API 密钥",
|
||||||
"ai_providers": "AI 提供商",
|
"ai_providers": "AI 提供商",
|
||||||
@@ -91,6 +92,27 @@
|
|||||||
"logs": "日志查看",
|
"logs": "日志查看",
|
||||||
"system_info": "中心信息"
|
"system_info": "中心信息"
|
||||||
},
|
},
|
||||||
|
"dashboard": {
|
||||||
|
"title": "仪表盘",
|
||||||
|
"subtitle": "欢迎使用 CLI Proxy API 管理中心",
|
||||||
|
"openai_providers": "OpenAI 提供商",
|
||||||
|
"quick_actions": "快捷操作",
|
||||||
|
"current_config": "当前配置",
|
||||||
|
"management_keys": "管理密钥",
|
||||||
|
"provider_keys_detail": "G:{{gemini}} C:{{codex}} Cl:{{claude}} O:{{openai}}",
|
||||||
|
"oauth_credentials": "OAuth 凭证",
|
||||||
|
"usage_overview": "使用概览",
|
||||||
|
"total_requests": "总请求数",
|
||||||
|
"total_tokens": "总 Token 数",
|
||||||
|
"rpm_30min": "RPM (30分钟)",
|
||||||
|
"tpm_30min": "TPM (30分钟)",
|
||||||
|
"models_used": "使用模型数",
|
||||||
|
"no_usage_data": "暂无使用数据",
|
||||||
|
"view_detailed_usage": "查看详细统计",
|
||||||
|
"edit_settings": "编辑设置",
|
||||||
|
"available_models": "可用模型",
|
||||||
|
"available_models_desc": "所有提供商的模型总数"
|
||||||
|
},
|
||||||
"basic_settings": {
|
"basic_settings": {
|
||||||
"title": "基础设置",
|
"title": "基础设置",
|
||||||
"debug_title": "调试模式",
|
"debug_title": "调试模式",
|
||||||
@@ -200,8 +222,6 @@
|
|||||||
"ampcode_upstream_api_key_current": "当前Amp官方密钥: {{key}}",
|
"ampcode_upstream_api_key_current": "当前Amp官方密钥: {{key}}",
|
||||||
"ampcode_clear_upstream_api_key": "清除官方密钥",
|
"ampcode_clear_upstream_api_key": "清除官方密钥",
|
||||||
"ampcode_clear_upstream_api_key_confirm": "确定要清除 Ampcode 的 upstream API key(Amp官方)吗?",
|
"ampcode_clear_upstream_api_key_confirm": "确定要清除 Ampcode 的 upstream API key(Amp官方)吗?",
|
||||||
"ampcode_restrict_management_label": "仅允许本机访问 Amp 管理路由",
|
|
||||||
"ampcode_restrict_management_hint": "开启后,/api/auth、/api/user、/api/threads 等 Amp 管理路由仅允许 127.0.0.1/::1 访问(推荐)。",
|
|
||||||
"ampcode_force_model_mappings_label": "强制应用模型映射",
|
"ampcode_force_model_mappings_label": "强制应用模型映射",
|
||||||
"ampcode_force_model_mappings_hint": "开启后,模型映射将覆盖本地 API Key 可用性判断。",
|
"ampcode_force_model_mappings_hint": "开启后,模型映射将覆盖本地 API Key 可用性判断。",
|
||||||
"ampcode_model_mappings_label": "模型映射 (from → to)",
|
"ampcode_model_mappings_label": "模型映射 (from → to)",
|
||||||
@@ -424,9 +444,10 @@
|
|||||||
"gemini_cli_oauth_title": "Gemini CLI OAuth",
|
"gemini_cli_oauth_title": "Gemini CLI OAuth",
|
||||||
"gemini_cli_oauth_button": "开始 Gemini CLI 登录",
|
"gemini_cli_oauth_button": "开始 Gemini CLI 登录",
|
||||||
"gemini_cli_oauth_hint": "通过 OAuth 流程登录 Google Gemini CLI 服务,自动获取并保存认证文件。",
|
"gemini_cli_oauth_hint": "通过 OAuth 流程登录 Google Gemini CLI 服务,自动获取并保存认证文件。",
|
||||||
"gemini_cli_project_id_label": "Google Cloud 项目 ID (可选):",
|
"gemini_cli_project_id_label": "Google Cloud 项目 ID:",
|
||||||
"gemini_cli_project_id_placeholder": "输入 Google Cloud 项目 ID (可选)",
|
"gemini_cli_project_id_placeholder": "输入 Google Cloud 项目 ID",
|
||||||
"gemini_cli_project_id_hint": "如果指定了项目 ID,将使用该项目的认证信息。",
|
"gemini_cli_project_id_hint": "请填写项目 ID,用于 Gemini CLI OAuth 登录。",
|
||||||
|
"gemini_cli_project_id_required": "请填写 Google Cloud 项目 ID。",
|
||||||
"gemini_cli_oauth_url_label": "授权链接:",
|
"gemini_cli_oauth_url_label": "授权链接:",
|
||||||
"gemini_cli_open_link": "打开链接",
|
"gemini_cli_open_link": "打开链接",
|
||||||
"gemini_cli_copy_link": "复制链接",
|
"gemini_cli_copy_link": "复制链接",
|
||||||
@@ -446,6 +467,16 @@
|
|||||||
"qwen_oauth_status_error": "认证失败:",
|
"qwen_oauth_status_error": "认证失败:",
|
||||||
"qwen_oauth_start_error": "启动 Qwen OAuth 失败:",
|
"qwen_oauth_start_error": "启动 Qwen OAuth 失败:",
|
||||||
"qwen_oauth_polling_error": "检查认证状态失败:",
|
"qwen_oauth_polling_error": "检查认证状态失败:",
|
||||||
|
"oauth_callback_label": "回调 URL",
|
||||||
|
"oauth_callback_placeholder": "http://localhost:1455/auth/callback?code=...&state=...",
|
||||||
|
"oauth_callback_hint": "远程浏览器模式:当授权跳转到 http://localhost:... 后,复制完整 URL 并提交到这里。",
|
||||||
|
"oauth_callback_button": "提交回调 URL",
|
||||||
|
"oauth_callback_required": "请先粘贴完整的回调 URL。",
|
||||||
|
"oauth_callback_success": "回调 URL 已提交,请继续等待认证。",
|
||||||
|
"oauth_callback_error": "提交回调 URL 失败:",
|
||||||
|
"oauth_callback_upgrade_hint": "请更新CLI Proxy API或检查连接",
|
||||||
|
"oauth_callback_status_success": "回调 URL 已提交,等待认证中...",
|
||||||
|
"oauth_callback_status_error": "回调 URL 提交失败:",
|
||||||
"missing_state": "无法获取认证状态参数",
|
"missing_state": "无法获取认证状态参数",
|
||||||
"iflow_oauth_title": "iFlow OAuth",
|
"iflow_oauth_title": "iFlow OAuth",
|
||||||
"iflow_oauth_button": "开始 iFlow 登录",
|
"iflow_oauth_button": "开始 iFlow 登录",
|
||||||
@@ -571,11 +602,15 @@
|
|||||||
"auto_refresh_disabled": "自动刷新已关闭",
|
"auto_refresh_disabled": "自动刷新已关闭",
|
||||||
"load_more_hint": "向上滚动加载更多",
|
"load_more_hint": "向上滚动加载更多",
|
||||||
"hidden_lines": "已隐藏 {{count}} 行",
|
"hidden_lines": "已隐藏 {{count}} 行",
|
||||||
|
"hide_management_logs": "屏蔽 {{prefix}} 日志",
|
||||||
"search_placeholder": "搜索日志内容或关键字",
|
"search_placeholder": "搜索日志内容或关键字",
|
||||||
"search_empty_title": "未找到匹配的日志",
|
"search_empty_title": "未找到匹配的日志",
|
||||||
"search_empty_desc": "尝试更换关键字或清空搜索条件。",
|
"search_empty_desc": "尝试更换关键字或清空筛选条件。",
|
||||||
|
"double_click_copy_hint": "双击复制日志原文",
|
||||||
|
"copy_success": "已复制日志原文",
|
||||||
|
"copy_failed": "复制失败",
|
||||||
"lines": "行",
|
"lines": "行",
|
||||||
"removed": "已删除",
|
"removed": "已过滤",
|
||||||
"upgrade_required_title": "需要升级 CLI Proxy API",
|
"upgrade_required_title": "需要升级 CLI Proxy API",
|
||||||
"upgrade_required_desc": "当前服务器版本不支持日志查看功能,请升级到最新版本的 CLI Proxy API 以使用此功能。"
|
"upgrade_required_desc": "当前服务器版本不支持日志查看功能,请升级到最新版本的 CLI Proxy API 以使用此功能。"
|
||||||
},
|
},
|
||||||
|
|||||||
@@ -404,8 +404,17 @@
|
|||||||
}
|
}
|
||||||
|
|
||||||
.excludedModelTag {
|
.excludedModelTag {
|
||||||
background: rgba(251, 191, 36, 0.2);
|
background: rgba(251, 191, 36, 0.22);
|
||||||
border-color: rgba(251, 191, 36, 0.4);
|
border-color: rgba(251, 191, 36, 0.55);
|
||||||
|
color: #fde68a;
|
||||||
|
|
||||||
|
.modelName {
|
||||||
|
color: #fde68a;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
.excludedModelsLabel {
|
||||||
|
color: #fde68a;
|
||||||
}
|
}
|
||||||
|
|
||||||
.apiKeyEntryCard {
|
.apiKeyEntryCard {
|
||||||
|
|||||||
@@ -49,7 +49,6 @@ interface OpenAIFormState {
|
|||||||
interface AmpcodeFormState {
|
interface AmpcodeFormState {
|
||||||
upstreamUrl: string;
|
upstreamUrl: string;
|
||||||
upstreamApiKey: string;
|
upstreamApiKey: string;
|
||||||
restrictManagementToLocalhost: boolean;
|
|
||||||
forceModelMappings: boolean;
|
forceModelMappings: boolean;
|
||||||
mappingEntries: ModelEntry[];
|
mappingEntries: ModelEntry[];
|
||||||
}
|
}
|
||||||
@@ -174,7 +173,6 @@ const entriesToAmpcodeMappings = (entries: ModelEntry[]): AmpcodeModelMapping[]
|
|||||||
const buildAmpcodeFormState = (ampcode?: AmpcodeConfig | null): AmpcodeFormState => ({
|
const buildAmpcodeFormState = (ampcode?: AmpcodeConfig | null): AmpcodeFormState => ({
|
||||||
upstreamUrl: ampcode?.upstreamUrl ?? '',
|
upstreamUrl: ampcode?.upstreamUrl ?? '',
|
||||||
upstreamApiKey: '',
|
upstreamApiKey: '',
|
||||||
restrictManagementToLocalhost: ampcode?.restrictManagementToLocalhost ?? true,
|
|
||||||
forceModelMappings: ampcode?.forceModelMappings ?? false,
|
forceModelMappings: ampcode?.forceModelMappings ?? false,
|
||||||
mappingEntries: ampcodeMappingsToEntries(ampcode?.modelMappings),
|
mappingEntries: ampcodeMappingsToEntries(ampcode?.modelMappings),
|
||||||
});
|
});
|
||||||
@@ -701,9 +699,6 @@ export function AiProvidersPage() {
|
|||||||
await ampcodeApi.clearUpstreamUrl();
|
await ampcodeApi.clearUpstreamUrl();
|
||||||
}
|
}
|
||||||
|
|
||||||
await ampcodeApi.updateRestrictManagementToLocalhost(
|
|
||||||
ampcodeForm.restrictManagementToLocalhost
|
|
||||||
);
|
|
||||||
await ampcodeApi.updateForceModelMappings(ampcodeForm.forceModelMappings);
|
await ampcodeApi.updateForceModelMappings(ampcodeForm.forceModelMappings);
|
||||||
|
|
||||||
if (ampcodeLoaded || ampcodeMappingsDirty) {
|
if (ampcodeLoaded || ampcodeMappingsDirty) {
|
||||||
@@ -720,12 +715,18 @@ export function AiProvidersPage() {
|
|||||||
|
|
||||||
const previous = config?.ampcode ?? {};
|
const previous = config?.ampcode ?? {};
|
||||||
const next: AmpcodeConfig = {
|
const next: AmpcodeConfig = {
|
||||||
...previous,
|
|
||||||
upstreamUrl: upstreamUrl || undefined,
|
upstreamUrl: upstreamUrl || undefined,
|
||||||
restrictManagementToLocalhost: ampcodeForm.restrictManagementToLocalhost,
|
|
||||||
forceModelMappings: ampcodeForm.forceModelMappings,
|
forceModelMappings: ampcodeForm.forceModelMappings,
|
||||||
};
|
};
|
||||||
|
|
||||||
|
if (previous.upstreamApiKey) {
|
||||||
|
next.upstreamApiKey = previous.upstreamApiKey;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (Array.isArray(previous.modelMappings)) {
|
||||||
|
next.modelMappings = previous.modelMappings;
|
||||||
|
}
|
||||||
|
|
||||||
if (overrideKey) {
|
if (overrideKey) {
|
||||||
next.upstreamApiKey = overrideKey;
|
next.upstreamApiKey = overrideKey;
|
||||||
}
|
}
|
||||||
@@ -1505,16 +1506,6 @@ export function AiProvidersPage() {
|
|||||||
: t('common.not_set')}
|
: t('common.not_set')}
|
||||||
</span>
|
</span>
|
||||||
</div>
|
</div>
|
||||||
<div className={styles.fieldRow}>
|
|
||||||
<span className={styles.fieldLabel}>
|
|
||||||
{t('ai_providers.ampcode_restrict_management_label')}:
|
|
||||||
</span>
|
|
||||||
<span className={styles.fieldValue}>
|
|
||||||
{(config?.ampcode?.restrictManagementToLocalhost ?? true)
|
|
||||||
? t('common.yes')
|
|
||||||
: t('common.no')}
|
|
||||||
</span>
|
|
||||||
</div>
|
|
||||||
<div className={styles.fieldRow}>
|
<div className={styles.fieldRow}>
|
||||||
<span className={styles.fieldLabel}>
|
<span className={styles.fieldLabel}>
|
||||||
{t('ai_providers.ampcode_force_model_mappings_label')}:
|
{t('ai_providers.ampcode_force_model_mappings_label')}:
|
||||||
@@ -1739,18 +1730,6 @@ export function AiProvidersPage() {
|
|||||||
</Button>
|
</Button>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div className="form-group">
|
|
||||||
<ToggleSwitch
|
|
||||||
label={t('ai_providers.ampcode_restrict_management_label')}
|
|
||||||
checked={ampcodeForm.restrictManagementToLocalhost}
|
|
||||||
onChange={(value) =>
|
|
||||||
setAmpcodeForm((prev) => ({ ...prev, restrictManagementToLocalhost: value }))
|
|
||||||
}
|
|
||||||
disabled={ampcodeModalLoading || ampcodeSaving}
|
|
||||||
/>
|
|
||||||
<div className="hint">{t('ai_providers.ampcode_restrict_management_hint')}</div>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div className="form-group">
|
<div className="form-group">
|
||||||
<ToggleSwitch
|
<ToggleSwitch
|
||||||
label={t('ai_providers.ampcode_force_model_mappings_label')}
|
label={t('ai_providers.ampcode_force_model_mappings_label')}
|
||||||
@@ -1897,19 +1876,21 @@ export function AiProvidersPage() {
|
|||||||
keyPlaceholder={t('common.custom_headers_key_placeholder')}
|
keyPlaceholder={t('common.custom_headers_key_placeholder')}
|
||||||
valuePlaceholder={t('common.custom_headers_value_placeholder')}
|
valuePlaceholder={t('common.custom_headers_value_placeholder')}
|
||||||
/>
|
/>
|
||||||
<div className="form-group">
|
{modal?.type === 'claude' && (
|
||||||
<label>{t('ai_providers.claude_models_label')}</label>
|
<div className="form-group">
|
||||||
<ModelInputList
|
<label>{t('ai_providers.claude_models_label')}</label>
|
||||||
entries={providerForm.modelEntries}
|
<ModelInputList
|
||||||
onChange={(entries) =>
|
entries={providerForm.modelEntries}
|
||||||
setProviderForm((prev) => ({ ...prev, modelEntries: entries }))
|
onChange={(entries) =>
|
||||||
}
|
setProviderForm((prev) => ({ ...prev, modelEntries: entries }))
|
||||||
addLabel={t('ai_providers.claude_models_add_btn')}
|
}
|
||||||
namePlaceholder={t('common.model_name_placeholder')}
|
addLabel={t('ai_providers.claude_models_add_btn')}
|
||||||
aliasPlaceholder={t('common.model_alias_placeholder')}
|
namePlaceholder={t('common.model_name_placeholder')}
|
||||||
disabled={saving}
|
aliasPlaceholder={t('common.model_alias_placeholder')}
|
||||||
/>
|
disabled={saving}
|
||||||
</div>
|
/>
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
<div className="form-group">
|
<div className="form-group">
|
||||||
<label>{t('ai_providers.excluded_models_label')}</label>
|
<label>{t('ai_providers.excluded_models_label')}</label>
|
||||||
<textarea
|
<textarea
|
||||||
|
|||||||
320
src/pages/DashboardPage.module.scss
Normal file
320
src/pages/DashboardPage.module.scss
Normal file
@@ -0,0 +1,320 @@
|
|||||||
|
@use 'sass:color';
|
||||||
|
@use '../styles/variables.scss' as *;
|
||||||
|
|
||||||
|
.dashboard {
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
gap: $spacing-lg;
|
||||||
|
max-width: 1000px;
|
||||||
|
margin: 0 auto;
|
||||||
|
}
|
||||||
|
|
||||||
|
.header {
|
||||||
|
margin-bottom: $spacing-sm;
|
||||||
|
}
|
||||||
|
|
||||||
|
.title {
|
||||||
|
font-size: 26px;
|
||||||
|
font-weight: 800;
|
||||||
|
color: var(--text-primary);
|
||||||
|
margin: 0;
|
||||||
|
line-height: 1.4;
|
||||||
|
}
|
||||||
|
|
||||||
|
.subtitle {
|
||||||
|
font-size: 15px;
|
||||||
|
color: var(--text-secondary);
|
||||||
|
margin: $spacing-xs 0 0 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
.connectionCard {
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
justify-content: space-between;
|
||||||
|
gap: $spacing-md;
|
||||||
|
background: var(--bg-secondary);
|
||||||
|
border: 1px solid var(--border-color);
|
||||||
|
border-radius: $radius-lg;
|
||||||
|
padding: $spacing-md $spacing-lg;
|
||||||
|
flex-wrap: wrap;
|
||||||
|
}
|
||||||
|
|
||||||
|
.connectionStatus {
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
gap: $spacing-sm;
|
||||||
|
}
|
||||||
|
|
||||||
|
.statusDot {
|
||||||
|
width: 10px;
|
||||||
|
height: 10px;
|
||||||
|
border-radius: 50%;
|
||||||
|
background: $gray-400;
|
||||||
|
|
||||||
|
&.connected {
|
||||||
|
background: $success-color;
|
||||||
|
box-shadow: 0 0 8px rgba($success-color, 0.5);
|
||||||
|
}
|
||||||
|
|
||||||
|
&.connecting {
|
||||||
|
background: $warning-color;
|
||||||
|
animation: pulse 1s ease-in-out infinite;
|
||||||
|
}
|
||||||
|
|
||||||
|
&.disconnected {
|
||||||
|
background: $error-color;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@keyframes pulse {
|
||||||
|
0%, 100% {
|
||||||
|
opacity: 1;
|
||||||
|
}
|
||||||
|
50% {
|
||||||
|
opacity: 0.5;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
.statusText {
|
||||||
|
font-weight: 600;
|
||||||
|
color: var(--text-primary);
|
||||||
|
}
|
||||||
|
|
||||||
|
.connectionInfo {
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
gap: $spacing-md;
|
||||||
|
flex-wrap: wrap;
|
||||||
|
}
|
||||||
|
|
||||||
|
.serverUrl {
|
||||||
|
font-family: $font-mono;
|
||||||
|
font-size: 13px;
|
||||||
|
color: var(--text-secondary);
|
||||||
|
background: var(--bg-primary);
|
||||||
|
padding: 4px 10px;
|
||||||
|
border-radius: $radius-md;
|
||||||
|
border: 1px solid var(--border-color);
|
||||||
|
}
|
||||||
|
|
||||||
|
.serverVersion {
|
||||||
|
font-size: 13px;
|
||||||
|
font-weight: 600;
|
||||||
|
color: var(--primary-color);
|
||||||
|
background: rgba($primary-color, 0.1);
|
||||||
|
padding: 4px 10px;
|
||||||
|
border-radius: $radius-full;
|
||||||
|
}
|
||||||
|
|
||||||
|
.buildDate {
|
||||||
|
font-size: 12px;
|
||||||
|
color: var(--text-secondary);
|
||||||
|
}
|
||||||
|
|
||||||
|
.statsGrid {
|
||||||
|
display: grid;
|
||||||
|
grid-template-columns: repeat(4, 1fr);
|
||||||
|
gap: $spacing-md;
|
||||||
|
|
||||||
|
@media (max-width: 900px) {
|
||||||
|
grid-template-columns: repeat(2, 1fr);
|
||||||
|
}
|
||||||
|
|
||||||
|
@media (max-width: 500px) {
|
||||||
|
grid-template-columns: 1fr;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
.statCard {
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
gap: $spacing-md;
|
||||||
|
padding: $spacing-lg;
|
||||||
|
background: var(--bg-primary);
|
||||||
|
border: 1px solid var(--border-color);
|
||||||
|
border-radius: $radius-lg;
|
||||||
|
text-decoration: none;
|
||||||
|
transition: all $transition-fast;
|
||||||
|
|
||||||
|
&:hover {
|
||||||
|
border-color: var(--primary-color);
|
||||||
|
box-shadow: 0 4px 12px rgba(0, 0, 0, 0.08);
|
||||||
|
transform: translateY(-2px);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
.statIcon {
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
justify-content: center;
|
||||||
|
width: 48px;
|
||||||
|
height: 48px;
|
||||||
|
border-radius: $radius-md;
|
||||||
|
background: var(--bg-secondary);
|
||||||
|
color: var(--primary-color);
|
||||||
|
}
|
||||||
|
|
||||||
|
.statContent {
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
gap: 2px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.statValue {
|
||||||
|
font-size: 24px;
|
||||||
|
font-weight: 800;
|
||||||
|
color: var(--text-primary);
|
||||||
|
}
|
||||||
|
|
||||||
|
.statLabel {
|
||||||
|
font-size: 13px;
|
||||||
|
color: var(--text-secondary);
|
||||||
|
}
|
||||||
|
|
||||||
|
.statSublabel {
|
||||||
|
font-size: 11px;
|
||||||
|
color: var(--text-secondary);
|
||||||
|
opacity: 0.8;
|
||||||
|
margin-top: 2px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.section {
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
gap: $spacing-md;
|
||||||
|
}
|
||||||
|
|
||||||
|
.sectionTitle {
|
||||||
|
font-size: 16px;
|
||||||
|
font-weight: 700;
|
||||||
|
color: var(--text-primary);
|
||||||
|
margin: 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
.actionsGrid {
|
||||||
|
display: flex;
|
||||||
|
flex-wrap: wrap;
|
||||||
|
gap: $spacing-sm;
|
||||||
|
|
||||||
|
a {
|
||||||
|
text-decoration: none;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
.actionButton {
|
||||||
|
display: inline-flex;
|
||||||
|
align-items: center;
|
||||||
|
gap: $spacing-sm;
|
||||||
|
|
||||||
|
// Button 内部的 span 需要 flex 对齐图标和文字
|
||||||
|
> span {
|
||||||
|
display: inline-flex;
|
||||||
|
align-items: center;
|
||||||
|
gap: $spacing-sm;
|
||||||
|
}
|
||||||
|
|
||||||
|
svg {
|
||||||
|
flex-shrink: 0;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
.configGrid {
|
||||||
|
display: grid;
|
||||||
|
grid-template-columns: repeat(auto-fit, minmax(200px, 1fr));
|
||||||
|
gap: $spacing-sm;
|
||||||
|
}
|
||||||
|
|
||||||
|
.configItem {
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
justify-content: space-between;
|
||||||
|
gap: $spacing-sm;
|
||||||
|
padding: $spacing-sm $spacing-md;
|
||||||
|
background: var(--bg-secondary);
|
||||||
|
border: 1px solid var(--border-color);
|
||||||
|
border-radius: $radius-md;
|
||||||
|
}
|
||||||
|
|
||||||
|
.configLabel {
|
||||||
|
font-size: 13px;
|
||||||
|
color: var(--text-secondary);
|
||||||
|
}
|
||||||
|
|
||||||
|
.configValue {
|
||||||
|
font-size: 13px;
|
||||||
|
font-weight: 600;
|
||||||
|
color: var(--text-primary);
|
||||||
|
|
||||||
|
&.enabled {
|
||||||
|
color: $success-color;
|
||||||
|
}
|
||||||
|
|
||||||
|
&.disabled {
|
||||||
|
color: var(--text-secondary);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
.configValueMono {
|
||||||
|
font-size: 12px;
|
||||||
|
font-family: $font-mono;
|
||||||
|
color: var(--text-secondary);
|
||||||
|
word-break: break-all;
|
||||||
|
}
|
||||||
|
|
||||||
|
.configItemFull {
|
||||||
|
grid-column: 1 / -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Usage stats section
|
||||||
|
.usageGrid {
|
||||||
|
display: grid;
|
||||||
|
grid-template-columns: repeat(auto-fit, minmax(140px, 1fr));
|
||||||
|
gap: $spacing-sm;
|
||||||
|
}
|
||||||
|
|
||||||
|
.usageCard {
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
align-items: center;
|
||||||
|
justify-content: center;
|
||||||
|
padding: $spacing-md;
|
||||||
|
background: var(--bg-secondary);
|
||||||
|
border: 1px solid var(--border-color);
|
||||||
|
border-radius: $radius-md;
|
||||||
|
text-align: center;
|
||||||
|
}
|
||||||
|
|
||||||
|
.usageValue {
|
||||||
|
font-size: 22px;
|
||||||
|
font-weight: 800;
|
||||||
|
color: var(--primary-color);
|
||||||
|
}
|
||||||
|
|
||||||
|
.usageLabel {
|
||||||
|
font-size: 12px;
|
||||||
|
color: var(--text-secondary);
|
||||||
|
margin-top: 4px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.usageLoading,
|
||||||
|
.usageEmpty {
|
||||||
|
padding: $spacing-lg;
|
||||||
|
text-align: center;
|
||||||
|
color: var(--text-secondary);
|
||||||
|
background: var(--bg-secondary);
|
||||||
|
border: 1px solid var(--border-color);
|
||||||
|
border-radius: $radius-md;
|
||||||
|
}
|
||||||
|
|
||||||
|
.viewMoreLink {
|
||||||
|
display: inline-flex;
|
||||||
|
align-items: center;
|
||||||
|
font-size: 13px;
|
||||||
|
color: var(--primary-color);
|
||||||
|
text-decoration: none;
|
||||||
|
margin-top: $spacing-xs;
|
||||||
|
|
||||||
|
&:hover {
|
||||||
|
text-decoration: underline;
|
||||||
|
}
|
||||||
|
}
|
||||||
321
src/pages/DashboardPage.tsx
Normal file
321
src/pages/DashboardPage.tsx
Normal file
@@ -0,0 +1,321 @@
|
|||||||
|
import { useCallback, useEffect, useRef, useState } from 'react';
|
||||||
|
import { Link } from 'react-router-dom';
|
||||||
|
import { useTranslation } from 'react-i18next';
|
||||||
|
import {
|
||||||
|
IconKey,
|
||||||
|
IconBot,
|
||||||
|
IconFileText,
|
||||||
|
IconSatellite
|
||||||
|
} from '@/components/ui/icons';
|
||||||
|
import { useAuthStore, useConfigStore, useModelsStore } from '@/stores';
|
||||||
|
import { apiKeysApi, providersApi, authFilesApi } from '@/services/api';
|
||||||
|
import styles from './DashboardPage.module.scss';
|
||||||
|
|
||||||
|
interface QuickStat {
|
||||||
|
label: string;
|
||||||
|
value: number | string;
|
||||||
|
icon: React.ReactNode;
|
||||||
|
path: string;
|
||||||
|
loading?: boolean;
|
||||||
|
sublabel?: string;
|
||||||
|
}
|
||||||
|
|
||||||
|
interface ProviderStats {
|
||||||
|
gemini: number | null;
|
||||||
|
codex: number | null;
|
||||||
|
claude: number | null;
|
||||||
|
openai: number | null;
|
||||||
|
}
|
||||||
|
|
||||||
|
export function DashboardPage() {
|
||||||
|
const { t, i18n } = useTranslation();
|
||||||
|
const connectionStatus = useAuthStore((state) => state.connectionStatus);
|
||||||
|
const serverVersion = useAuthStore((state) => state.serverVersion);
|
||||||
|
const serverBuildDate = useAuthStore((state) => state.serverBuildDate);
|
||||||
|
const apiBase = useAuthStore((state) => state.apiBase);
|
||||||
|
const config = useConfigStore((state) => state.config);
|
||||||
|
|
||||||
|
const models = useModelsStore((state) => state.models);
|
||||||
|
const modelsLoading = useModelsStore((state) => state.loading);
|
||||||
|
const fetchModelsFromStore = useModelsStore((state) => state.fetchModels);
|
||||||
|
|
||||||
|
const [stats, setStats] = useState<{
|
||||||
|
apiKeys: number | null;
|
||||||
|
authFiles: number | null;
|
||||||
|
}>({
|
||||||
|
apiKeys: null,
|
||||||
|
authFiles: null
|
||||||
|
});
|
||||||
|
|
||||||
|
const [providerStats, setProviderStats] = useState<ProviderStats>({
|
||||||
|
gemini: null,
|
||||||
|
codex: null,
|
||||||
|
claude: null,
|
||||||
|
openai: null
|
||||||
|
});
|
||||||
|
|
||||||
|
const [loading, setLoading] = useState(true);
|
||||||
|
|
||||||
|
const apiKeysCache = useRef<string[]>([]);
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
apiKeysCache.current = [];
|
||||||
|
}, [apiBase, config?.apiKeys]);
|
||||||
|
|
||||||
|
const normalizeApiKeyList = (input: any): string[] => {
|
||||||
|
if (!Array.isArray(input)) return [];
|
||||||
|
const seen = new Set<string>();
|
||||||
|
const keys: string[] = [];
|
||||||
|
|
||||||
|
input.forEach((item) => {
|
||||||
|
const value = typeof item === 'string' ? item : item?.['api-key'] ?? item?.apiKey ?? '';
|
||||||
|
const trimmed = String(value || '').trim();
|
||||||
|
if (!trimmed || seen.has(trimmed)) return;
|
||||||
|
seen.add(trimmed);
|
||||||
|
keys.push(trimmed);
|
||||||
|
});
|
||||||
|
|
||||||
|
return keys;
|
||||||
|
};
|
||||||
|
|
||||||
|
const resolveApiKeysForModels = useCallback(async () => {
|
||||||
|
if (apiKeysCache.current.length) {
|
||||||
|
return apiKeysCache.current;
|
||||||
|
}
|
||||||
|
|
||||||
|
const configKeys = normalizeApiKeyList(config?.apiKeys);
|
||||||
|
if (configKeys.length) {
|
||||||
|
apiKeysCache.current = configKeys;
|
||||||
|
return configKeys;
|
||||||
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
|
const list = await apiKeysApi.list();
|
||||||
|
const normalized = normalizeApiKeyList(list);
|
||||||
|
if (normalized.length) {
|
||||||
|
apiKeysCache.current = normalized;
|
||||||
|
}
|
||||||
|
return normalized;
|
||||||
|
} catch {
|
||||||
|
return [];
|
||||||
|
}
|
||||||
|
}, [config?.apiKeys]);
|
||||||
|
|
||||||
|
const fetchModels = useCallback(async () => {
|
||||||
|
if (connectionStatus !== 'connected' || !apiBase) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
|
const apiKeys = await resolveApiKeysForModels();
|
||||||
|
const primaryKey = apiKeys[0];
|
||||||
|
await fetchModelsFromStore(apiBase, primaryKey);
|
||||||
|
} catch {
|
||||||
|
// Ignore model fetch errors on dashboard
|
||||||
|
}
|
||||||
|
}, [connectionStatus, apiBase, resolveApiKeysForModels, fetchModelsFromStore]);
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
const fetchStats = async () => {
|
||||||
|
setLoading(true);
|
||||||
|
try {
|
||||||
|
const [keysRes, filesRes, geminiRes, codexRes, claudeRes, openaiRes] = await Promise.allSettled([
|
||||||
|
apiKeysApi.list(),
|
||||||
|
authFilesApi.list(),
|
||||||
|
providersApi.getGeminiKeys(),
|
||||||
|
providersApi.getCodexConfigs(),
|
||||||
|
providersApi.getClaudeConfigs(),
|
||||||
|
providersApi.getOpenAIProviders()
|
||||||
|
]);
|
||||||
|
|
||||||
|
setStats({
|
||||||
|
apiKeys: keysRes.status === 'fulfilled' ? keysRes.value.length : null,
|
||||||
|
authFiles: filesRes.status === 'fulfilled' ? filesRes.value.files.length : null
|
||||||
|
});
|
||||||
|
|
||||||
|
setProviderStats({
|
||||||
|
gemini: geminiRes.status === 'fulfilled' ? geminiRes.value.length : null,
|
||||||
|
codex: codexRes.status === 'fulfilled' ? codexRes.value.length : null,
|
||||||
|
claude: claudeRes.status === 'fulfilled' ? claudeRes.value.length : null,
|
||||||
|
openai: openaiRes.status === 'fulfilled' ? openaiRes.value.length : null
|
||||||
|
});
|
||||||
|
} finally {
|
||||||
|
setLoading(false);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
if (connectionStatus === 'connected') {
|
||||||
|
fetchStats();
|
||||||
|
fetchModels();
|
||||||
|
} else {
|
||||||
|
setLoading(false);
|
||||||
|
}
|
||||||
|
}, [connectionStatus, fetchModels]);
|
||||||
|
|
||||||
|
// Calculate total provider keys only when all provider stats are available.
|
||||||
|
const providerStatsReady =
|
||||||
|
providerStats.gemini !== null &&
|
||||||
|
providerStats.codex !== null &&
|
||||||
|
providerStats.claude !== null &&
|
||||||
|
providerStats.openai !== null;
|
||||||
|
const hasProviderStats =
|
||||||
|
providerStats.gemini !== null ||
|
||||||
|
providerStats.codex !== null ||
|
||||||
|
providerStats.claude !== null ||
|
||||||
|
providerStats.openai !== null;
|
||||||
|
const totalProviderKeys = providerStatsReady
|
||||||
|
? (providerStats.gemini ?? 0) +
|
||||||
|
(providerStats.codex ?? 0) +
|
||||||
|
(providerStats.claude ?? 0) +
|
||||||
|
(providerStats.openai ?? 0)
|
||||||
|
: 0;
|
||||||
|
|
||||||
|
const quickStats: QuickStat[] = [
|
||||||
|
{
|
||||||
|
label: t('nav.api_keys'),
|
||||||
|
value: stats.apiKeys ?? '-',
|
||||||
|
icon: <IconKey size={24} />,
|
||||||
|
path: '/api-keys',
|
||||||
|
loading: loading && stats.apiKeys === null,
|
||||||
|
sublabel: t('dashboard.management_keys')
|
||||||
|
},
|
||||||
|
{
|
||||||
|
label: t('nav.ai_providers'),
|
||||||
|
value: loading ? '-' : providerStatsReady ? totalProviderKeys : '-',
|
||||||
|
icon: <IconBot size={24} />,
|
||||||
|
path: '/ai-providers',
|
||||||
|
loading: loading,
|
||||||
|
sublabel: hasProviderStats
|
||||||
|
? t('dashboard.provider_keys_detail', {
|
||||||
|
gemini: providerStats.gemini ?? '-',
|
||||||
|
codex: providerStats.codex ?? '-',
|
||||||
|
claude: providerStats.claude ?? '-',
|
||||||
|
openai: providerStats.openai ?? '-'
|
||||||
|
})
|
||||||
|
: undefined
|
||||||
|
},
|
||||||
|
{
|
||||||
|
label: t('nav.auth_files'),
|
||||||
|
value: stats.authFiles ?? '-',
|
||||||
|
icon: <IconFileText size={24} />,
|
||||||
|
path: '/auth-files',
|
||||||
|
loading: loading && stats.authFiles === null,
|
||||||
|
sublabel: t('dashboard.oauth_credentials')
|
||||||
|
},
|
||||||
|
{
|
||||||
|
label: t('dashboard.available_models'),
|
||||||
|
value: modelsLoading ? '-' : models.length,
|
||||||
|
icon: <IconSatellite size={24} />,
|
||||||
|
path: '/system',
|
||||||
|
loading: modelsLoading,
|
||||||
|
sublabel: t('dashboard.available_models_desc')
|
||||||
|
}
|
||||||
|
];
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div className={styles.dashboard}>
|
||||||
|
<div className={styles.header}>
|
||||||
|
<h1 className={styles.title}>{t('dashboard.title')}</h1>
|
||||||
|
<p className={styles.subtitle}>{t('dashboard.subtitle')}</p>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div className={styles.connectionCard}>
|
||||||
|
<div className={styles.connectionStatus}>
|
||||||
|
<span
|
||||||
|
className={`${styles.statusDot} ${
|
||||||
|
connectionStatus === 'connected'
|
||||||
|
? styles.connected
|
||||||
|
: connectionStatus === 'connecting'
|
||||||
|
? styles.connecting
|
||||||
|
: styles.disconnected
|
||||||
|
}`}
|
||||||
|
/>
|
||||||
|
<span className={styles.statusText}>
|
||||||
|
{t(
|
||||||
|
connectionStatus === 'connected'
|
||||||
|
? 'common.connected'
|
||||||
|
: connectionStatus === 'connecting'
|
||||||
|
? 'common.connecting'
|
||||||
|
: 'common.disconnected'
|
||||||
|
)}
|
||||||
|
</span>
|
||||||
|
</div>
|
||||||
|
<div className={styles.connectionInfo}>
|
||||||
|
<span className={styles.serverUrl}>{apiBase || '-'}</span>
|
||||||
|
{serverVersion && <span className={styles.serverVersion}>v{serverVersion}</span>}
|
||||||
|
{serverBuildDate && (
|
||||||
|
<span className={styles.buildDate}>
|
||||||
|
{new Date(serverBuildDate).toLocaleDateString(i18n.language)}
|
||||||
|
</span>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div className={styles.statsGrid}>
|
||||||
|
{quickStats.map((stat) => (
|
||||||
|
<Link key={stat.path} to={stat.path} className={styles.statCard}>
|
||||||
|
<div className={styles.statIcon}>{stat.icon}</div>
|
||||||
|
<div className={styles.statContent}>
|
||||||
|
<span className={styles.statValue}>{stat.loading ? '...' : stat.value}</span>
|
||||||
|
<span className={styles.statLabel}>{stat.label}</span>
|
||||||
|
{stat.sublabel && !stat.loading && (
|
||||||
|
<span className={styles.statSublabel}>{stat.sublabel}</span>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
</Link>
|
||||||
|
))}
|
||||||
|
</div>
|
||||||
|
|
||||||
|
{config && (
|
||||||
|
<div className={styles.section}>
|
||||||
|
<h2 className={styles.sectionTitle}>{t('dashboard.current_config')}</h2>
|
||||||
|
<div className={styles.configGrid}>
|
||||||
|
<div className={styles.configItem}>
|
||||||
|
<span className={styles.configLabel}>{t('basic_settings.debug_enable')}</span>
|
||||||
|
<span className={`${styles.configValue} ${config.debug ? styles.enabled : styles.disabled}`}>
|
||||||
|
{config.debug ? t('common.yes') : t('common.no')}
|
||||||
|
</span>
|
||||||
|
</div>
|
||||||
|
<div className={styles.configItem}>
|
||||||
|
<span className={styles.configLabel}>{t('basic_settings.usage_statistics_enable')}</span>
|
||||||
|
<span className={`${styles.configValue} ${config.usageStatisticsEnabled ? styles.enabled : styles.disabled}`}>
|
||||||
|
{config.usageStatisticsEnabled ? t('common.yes') : t('common.no')}
|
||||||
|
</span>
|
||||||
|
</div>
|
||||||
|
<div className={styles.configItem}>
|
||||||
|
<span className={styles.configLabel}>{t('basic_settings.logging_to_file_enable')}</span>
|
||||||
|
<span className={`${styles.configValue} ${config.loggingToFile ? styles.enabled : styles.disabled}`}>
|
||||||
|
{config.loggingToFile ? t('common.yes') : t('common.no')}
|
||||||
|
</span>
|
||||||
|
</div>
|
||||||
|
<div className={styles.configItem}>
|
||||||
|
<span className={styles.configLabel}>{t('basic_settings.request_log_enable')}</span>
|
||||||
|
<span className={`${styles.configValue} ${config.requestLog ? styles.enabled : styles.disabled}`}>
|
||||||
|
{config.requestLog ? t('common.yes') : t('common.no')}
|
||||||
|
</span>
|
||||||
|
</div>
|
||||||
|
<div className={styles.configItem}>
|
||||||
|
<span className={styles.configLabel}>{t('basic_settings.retry_count_label')}</span>
|
||||||
|
<span className={styles.configValue}>{config.requestRetry ?? 0}</span>
|
||||||
|
</div>
|
||||||
|
<div className={styles.configItem}>
|
||||||
|
<span className={styles.configLabel}>{t('basic_settings.ws_auth_enable')}</span>
|
||||||
|
<span className={`${styles.configValue} ${config.wsAuth ? styles.enabled : styles.disabled}`}>
|
||||||
|
{config.wsAuth ? t('common.yes') : t('common.no')}
|
||||||
|
</span>
|
||||||
|
</div>
|
||||||
|
{config.proxyUrl && (
|
||||||
|
<div className={`${styles.configItem} ${styles.configItemFull}`}>
|
||||||
|
<span className={styles.configLabel}>{t('basic_settings.proxy_url_label')}</span>
|
||||||
|
<span className={styles.configValueMono}>{config.proxyUrl}</span>
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
<Link to="/settings" className={styles.viewMoreLink}>
|
||||||
|
{t('dashboard.edit_settings')} →
|
||||||
|
</Link>
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
}
|
||||||
@@ -28,6 +28,62 @@
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.filters {
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
gap: $spacing-md;
|
||||||
|
flex-wrap: wrap;
|
||||||
|
margin-bottom: $spacing-md;
|
||||||
|
|
||||||
|
:global(.form-group) {
|
||||||
|
margin: 0;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
.searchWrapper {
|
||||||
|
flex: 1;
|
||||||
|
min-width: 220px;
|
||||||
|
max-width: 420px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.searchInput {
|
||||||
|
padding-right: 44px !important;
|
||||||
|
}
|
||||||
|
|
||||||
|
.searchIcon {
|
||||||
|
color: var(--text-tertiary);
|
||||||
|
pointer-events: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
.searchClear {
|
||||||
|
@include button-reset;
|
||||||
|
display: inline-flex;
|
||||||
|
align-items: center;
|
||||||
|
justify-content: center;
|
||||||
|
width: 28px;
|
||||||
|
height: 28px;
|
||||||
|
border-radius: $radius-full;
|
||||||
|
color: var(--text-secondary);
|
||||||
|
|
||||||
|
&:hover {
|
||||||
|
background: var(--bg-secondary);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
.filterStats {
|
||||||
|
margin-left: auto;
|
||||||
|
display: inline-flex;
|
||||||
|
align-items: center;
|
||||||
|
gap: 10px;
|
||||||
|
font-size: 12px;
|
||||||
|
color: var(--text-secondary);
|
||||||
|
white-space: nowrap;
|
||||||
|
}
|
||||||
|
|
||||||
|
.removedCount {
|
||||||
|
color: var(--text-tertiary);
|
||||||
|
}
|
||||||
|
|
||||||
.actionButton {
|
.actionButton {
|
||||||
white-space: nowrap;
|
white-space: nowrap;
|
||||||
}
|
}
|
||||||
@@ -93,7 +149,9 @@
|
|||||||
padding: 10px 12px;
|
padding: 10px 12px;
|
||||||
border-bottom: 1px solid var(--border-color);
|
border-bottom: 1px solid var(--border-color);
|
||||||
border-left: 3px solid transparent;
|
border-left: 3px solid transparent;
|
||||||
font-family: ui-monospace, SFMono-Regular, Menlo, Monaco, Consolas, 'Liberation Mono', 'Courier New',
|
cursor: copy;
|
||||||
|
font-family:
|
||||||
|
ui-monospace, SFMono-Regular, Menlo, Monaco, Consolas, 'Liberation Mono', 'Courier New',
|
||||||
monospace;
|
monospace;
|
||||||
font-size: 12.5px;
|
font-size: 12.5px;
|
||||||
line-height: 1.45;
|
line-height: 1.45;
|
||||||
|
|||||||
@@ -1,12 +1,22 @@
|
|||||||
import { useEffect, useLayoutEffect, useMemo, useRef, useState } from 'react';
|
import { useDeferredValue, useEffect, useLayoutEffect, useMemo, useRef, useState } from 'react';
|
||||||
import { useTranslation } from 'react-i18next';
|
import { useTranslation } from 'react-i18next';
|
||||||
import { Card } from '@/components/ui/Card';
|
import { Card } from '@/components/ui/Card';
|
||||||
import { Button } from '@/components/ui/Button';
|
import { Button } from '@/components/ui/Button';
|
||||||
import { EmptyState } from '@/components/ui/EmptyState';
|
import { EmptyState } from '@/components/ui/EmptyState';
|
||||||
|
import { Input } from '@/components/ui/Input';
|
||||||
import { ToggleSwitch } from '@/components/ui/ToggleSwitch';
|
import { ToggleSwitch } from '@/components/ui/ToggleSwitch';
|
||||||
import { IconDownload, IconRefreshCw, IconTimer, IconTrash2 } from '@/components/ui/icons';
|
import {
|
||||||
|
IconDownload,
|
||||||
|
IconEyeOff,
|
||||||
|
IconRefreshCw,
|
||||||
|
IconSearch,
|
||||||
|
IconTimer,
|
||||||
|
IconTrash2,
|
||||||
|
IconX,
|
||||||
|
} from '@/components/ui/icons';
|
||||||
import { useNotificationStore, useAuthStore } from '@/stores';
|
import { useNotificationStore, useAuthStore } from '@/stores';
|
||||||
import { logsApi } from '@/services/api/logs';
|
import { logsApi } from '@/services/api/logs';
|
||||||
|
import { MANAGEMENT_API_PREFIX } from '@/utils/constants';
|
||||||
import { formatUnixTimestamp } from '@/utils/format';
|
import { formatUnixTimestamp } from '@/utils/format';
|
||||||
import styles from './LogsPage.module.scss';
|
import styles from './LogsPage.module.scss';
|
||||||
|
|
||||||
@@ -40,13 +50,15 @@ const LOG_LATENCY_REGEX = /\b(\d+(?:\.\d+)?)(?:\s*)(µs|us|ms|s)\b/i;
|
|||||||
const LOG_IPV4_REGEX = /\b(?:\d{1,3}\.){3}\d{1,3}\b/;
|
const LOG_IPV4_REGEX = /\b(?:\d{1,3}\.){3}\d{1,3}\b/;
|
||||||
const LOG_IPV6_REGEX = /\b(?:[a-f0-9]{0,4}:){2,7}[a-f0-9]{0,4}\b/i;
|
const LOG_IPV6_REGEX = /\b(?:[a-f0-9]{0,4}:){2,7}[a-f0-9]{0,4}\b/i;
|
||||||
const LOG_TIME_OF_DAY_REGEX = /^\d{1,2}:\d{2}:\d{2}(?:\.\d{1,3})?$/;
|
const LOG_TIME_OF_DAY_REGEX = /^\d{1,2}:\d{2}:\d{2}(?:\.\d{1,3})?$/;
|
||||||
|
const GIN_TIMESTAMP_SEGMENT_REGEX =
|
||||||
|
/^\[GIN\]\s+(\d{4})\/(\d{2})\/(\d{2})\s*-\s*(\d{2}:\d{2}:\d{2}(?:\.\d{1,3})?)\s*$/;
|
||||||
|
|
||||||
const HTTP_STATUS_PATTERNS: RegExp[] = [
|
const HTTP_STATUS_PATTERNS: RegExp[] = [
|
||||||
/\|\s*([1-5]\d{2})\s*\|/,
|
/\|\s*([1-5]\d{2})\s*\|/,
|
||||||
/\b([1-5]\d{2})\s*-/,
|
/\b([1-5]\d{2})\s*-/,
|
||||||
new RegExp(`\\b(?:${HTTP_METHODS.join('|')})\\s+\\S+\\s+([1-5]\\d{2})\\b`),
|
new RegExp(`\\b(?:${HTTP_METHODS.join('|')})\\s+\\S+\\s+([1-5]\\d{2})\\b`),
|
||||||
/\b(?:status|code|http)[:\s]+([1-5]\d{2})\b/i,
|
/\b(?:status|code|http)[:\s]+([1-5]\d{2})\b/i,
|
||||||
/\b([1-5]\d{2})\s+(?:OK|Created|Accepted|No Content|Moved|Found|Bad Request|Unauthorized|Forbidden|Not Found|Method Not Allowed|Internal Server Error|Bad Gateway|Service Unavailable|Gateway Timeout)\b/i
|
/\b([1-5]\d{2})\s+(?:OK|Created|Accepted|No Content|Moved|Found|Bad Request|Unauthorized|Forbidden|Not Found|Method Not Allowed|Internal Server Error|Bad Gateway|Service Unavailable|Gateway Timeout)\b/i,
|
||||||
];
|
];
|
||||||
|
|
||||||
const detectHttpStatusCode = (text: string): number | undefined => {
|
const detectHttpStatusCode = (text: string): number | undefined => {
|
||||||
@@ -78,6 +90,13 @@ const extractIp = (text: string): string | undefined => {
|
|||||||
return candidate;
|
return candidate;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
const normalizeTimestampToSeconds = (value: string): string => {
|
||||||
|
const trimmed = value.trim();
|
||||||
|
const match = trimmed.match(/^(\d{4}-\d{2}-\d{2})[ T](\d{2}:\d{2}:\d{2})/);
|
||||||
|
if (!match) return trimmed;
|
||||||
|
return `${match[1]} ${match[2]}`;
|
||||||
|
};
|
||||||
|
|
||||||
type ParsedLogLine = {
|
type ParsedLogLine = {
|
||||||
raw: string;
|
raw: string;
|
||||||
timestamp?: string;
|
timestamp?: string;
|
||||||
@@ -163,6 +182,23 @@ const parseLogLine = (raw: string): ParsedLogLine => {
|
|||||||
.filter(Boolean);
|
.filter(Boolean);
|
||||||
const consumed = new Set<number>();
|
const consumed = new Set<number>();
|
||||||
|
|
||||||
|
const ginIndex = segments.findIndex((segment) => GIN_TIMESTAMP_SEGMENT_REGEX.test(segment));
|
||||||
|
if (ginIndex >= 0) {
|
||||||
|
const match = segments[ginIndex].match(GIN_TIMESTAMP_SEGMENT_REGEX);
|
||||||
|
if (match) {
|
||||||
|
const ginTimestamp = `${match[1]}-${match[2]}-${match[3]} ${match[4]}`;
|
||||||
|
const normalizedGin = normalizeTimestampToSeconds(ginTimestamp);
|
||||||
|
const normalizedParsed = timestamp ? normalizeTimestampToSeconds(timestamp) : undefined;
|
||||||
|
|
||||||
|
if (!timestamp) {
|
||||||
|
timestamp = ginTimestamp;
|
||||||
|
consumed.add(ginIndex);
|
||||||
|
} else if (normalizedParsed === normalizedGin) {
|
||||||
|
consumed.add(ginIndex);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
// status code
|
// status code
|
||||||
const statusIndex = segments.findIndex((segment) => /^\d{3}\b/.test(segment));
|
const statusIndex = segments.findIndex((segment) => /^\d{3}\b/.test(segment));
|
||||||
if (statusIndex >= 0) {
|
if (statusIndex >= 0) {
|
||||||
@@ -187,9 +223,7 @@ const parseLogLine = (raw: string): ParsedLogLine => {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// ip
|
// ip
|
||||||
const ipIndex = segments.findIndex(
|
const ipIndex = segments.findIndex((segment) => Boolean(extractIp(segment)));
|
||||||
(segment) => Boolean(extractIp(segment))
|
|
||||||
);
|
|
||||||
if (ipIndex >= 0) {
|
if (ipIndex >= 0) {
|
||||||
const extracted = extractIp(segments[ipIndex]);
|
const extracted = extractIp(segments[ipIndex]);
|
||||||
if (extracted) {
|
if (extracted) {
|
||||||
@@ -226,6 +260,17 @@ const parseLogLine = (raw: string): ParsedLogLine => {
|
|||||||
|
|
||||||
if (!level) level = inferLogLevel(raw);
|
if (!level) level = inferLogLevel(raw);
|
||||||
|
|
||||||
|
if (message) {
|
||||||
|
const match = message.match(GIN_TIMESTAMP_SEGMENT_REGEX);
|
||||||
|
if (match) {
|
||||||
|
const ginTimestamp = `${match[1]}-${match[2]}-${match[3]} ${match[4]}`;
|
||||||
|
if (!timestamp) timestamp = ginTimestamp;
|
||||||
|
if (normalizeTimestampToSeconds(timestamp) === normalizeTimestampToSeconds(ginTimestamp)) {
|
||||||
|
message = '';
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
return {
|
return {
|
||||||
raw,
|
raw,
|
||||||
timestamp,
|
timestamp,
|
||||||
@@ -236,10 +281,44 @@ const parseLogLine = (raw: string): ParsedLogLine => {
|
|||||||
ip,
|
ip,
|
||||||
method,
|
method,
|
||||||
path,
|
path,
|
||||||
message
|
message,
|
||||||
};
|
};
|
||||||
};
|
};
|
||||||
|
|
||||||
|
const getErrorMessage = (err: unknown): string => {
|
||||||
|
if (err instanceof Error) return err.message;
|
||||||
|
if (typeof err === 'string') return err;
|
||||||
|
if (typeof err !== 'object' || err === null) return '';
|
||||||
|
if (!('message' in err)) return '';
|
||||||
|
|
||||||
|
const message = (err as { message?: unknown }).message;
|
||||||
|
return typeof message === 'string' ? message : '';
|
||||||
|
};
|
||||||
|
|
||||||
|
const copyToClipboard = async (text: string) => {
|
||||||
|
try {
|
||||||
|
await navigator.clipboard.writeText(text);
|
||||||
|
return true;
|
||||||
|
} catch {
|
||||||
|
try {
|
||||||
|
const textarea = document.createElement('textarea');
|
||||||
|
textarea.value = text;
|
||||||
|
textarea.style.position = 'fixed';
|
||||||
|
textarea.style.opacity = '0';
|
||||||
|
textarea.style.left = '-9999px';
|
||||||
|
textarea.style.top = '0';
|
||||||
|
document.body.appendChild(textarea);
|
||||||
|
textarea.focus();
|
||||||
|
textarea.select();
|
||||||
|
const ok = document.execCommand('copy');
|
||||||
|
document.body.removeChild(textarea);
|
||||||
|
return ok;
|
||||||
|
} catch {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
export function LogsPage() {
|
export function LogsPage() {
|
||||||
const { t } = useTranslation();
|
const { t } = useTranslation();
|
||||||
const { showNotification } = useNotificationStore();
|
const { showNotification } = useNotificationStore();
|
||||||
@@ -249,6 +328,9 @@ export function LogsPage() {
|
|||||||
const [loading, setLoading] = useState(true);
|
const [loading, setLoading] = useState(true);
|
||||||
const [error, setError] = useState('');
|
const [error, setError] = useState('');
|
||||||
const [autoRefresh, setAutoRefresh] = useState(false);
|
const [autoRefresh, setAutoRefresh] = useState(false);
|
||||||
|
const [searchQuery, setSearchQuery] = useState('');
|
||||||
|
const deferredSearchQuery = useDeferredValue(searchQuery);
|
||||||
|
const [hideManagementLogs, setHideManagementLogs] = useState(false);
|
||||||
const [errorLogs, setErrorLogs] = useState<ErrorLogItem[]>([]);
|
const [errorLogs, setErrorLogs] = useState<ErrorLogItem[]>([]);
|
||||||
const [loadingErrors, setLoadingErrors] = useState(false);
|
const [loadingErrors, setLoadingErrors] = useState(false);
|
||||||
|
|
||||||
@@ -287,9 +369,8 @@ export function LogsPage() {
|
|||||||
try {
|
try {
|
||||||
pendingScrollToBottomRef.current = !incremental || isNearBottom(logViewerRef.current);
|
pendingScrollToBottomRef.current = !incremental || isNearBottom(logViewerRef.current);
|
||||||
|
|
||||||
const params = incremental && latestTimestampRef.current > 0
|
const params =
|
||||||
? { after: latestTimestampRef.current }
|
incremental && latestTimestampRef.current > 0 ? { after: latestTimestampRef.current } : {};
|
||||||
: {};
|
|
||||||
const data = await logsApi.fetchLogs(params);
|
const data = await logsApi.fetchLogs(params);
|
||||||
|
|
||||||
// 更新时间戳
|
// 更新时间戳
|
||||||
@@ -321,10 +402,10 @@ export function LogsPage() {
|
|||||||
const visibleFrom = Math.max(buffer.length - INITIAL_DISPLAY_LINES, 0);
|
const visibleFrom = Math.max(buffer.length - INITIAL_DISPLAY_LINES, 0);
|
||||||
setLogState({ buffer, visibleFrom });
|
setLogState({ buffer, visibleFrom });
|
||||||
}
|
}
|
||||||
} catch (err: any) {
|
} catch (err: unknown) {
|
||||||
console.error('Failed to load logs:', err);
|
console.error('Failed to load logs:', err);
|
||||||
if (!incremental) {
|
if (!incremental) {
|
||||||
setError(err?.message || t('logs.load_error'));
|
setError(getErrorMessage(err) || t('logs.load_error'));
|
||||||
}
|
}
|
||||||
} finally {
|
} finally {
|
||||||
if (!incremental) {
|
if (!incremental) {
|
||||||
@@ -340,8 +421,12 @@ export function LogsPage() {
|
|||||||
setLogState({ buffer: [], visibleFrom: 0 });
|
setLogState({ buffer: [], visibleFrom: 0 });
|
||||||
latestTimestampRef.current = 0;
|
latestTimestampRef.current = 0;
|
||||||
showNotification(t('logs.clear_success'), 'success');
|
showNotification(t('logs.clear_success'), 'success');
|
||||||
} catch (err: any) {
|
} catch (err: unknown) {
|
||||||
showNotification(`${t('notification.delete_failed')}: ${err?.message || ''}`, 'error');
|
const message = getErrorMessage(err);
|
||||||
|
showNotification(
|
||||||
|
`${t('notification.delete_failed')}${message ? `: ${message}` : ''}`,
|
||||||
|
'error'
|
||||||
|
);
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
@@ -367,16 +452,8 @@ export function LogsPage() {
|
|||||||
try {
|
try {
|
||||||
const res = await logsApi.fetchErrorLogs();
|
const res = await logsApi.fetchErrorLogs();
|
||||||
// API 返回 { files: [...] }
|
// API 返回 { files: [...] }
|
||||||
const files = (res as any)?.files;
|
setErrorLogs(Array.isArray(res.files) ? res.files : []);
|
||||||
const list: ErrorLogItem[] = Array.isArray(files)
|
} catch (err: unknown) {
|
||||||
? files.map((f: any) => ({
|
|
||||||
name: f.name,
|
|
||||||
size: f.size,
|
|
||||||
modified: f.modified
|
|
||||||
}))
|
|
||||||
: [];
|
|
||||||
setErrorLogs(list);
|
|
||||||
} catch (err: any) {
|
|
||||||
console.error('Failed to load error logs:', err);
|
console.error('Failed to load error logs:', err);
|
||||||
// 静默失败,不影响主日志显示
|
// 静默失败,不影响主日志显示
|
||||||
setErrorLogs([]);
|
setErrorLogs([]);
|
||||||
@@ -396,8 +473,12 @@ export function LogsPage() {
|
|||||||
a.click();
|
a.click();
|
||||||
window.URL.revokeObjectURL(url);
|
window.URL.revokeObjectURL(url);
|
||||||
showNotification(t('logs.error_log_download_success'), 'success');
|
showNotification(t('logs.error_log_download_success'), 'success');
|
||||||
} catch (err: any) {
|
} catch (err: unknown) {
|
||||||
showNotification(`${t('notification.download_failed')}: ${err?.message || ''}`, 'error');
|
const message = getErrorMessage(err);
|
||||||
|
showNotification(
|
||||||
|
`${t('notification.download_failed')}${message ? `: ${message}` : ''}`,
|
||||||
|
'error'
|
||||||
|
);
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
@@ -434,23 +515,65 @@ export function LogsPage() {
|
|||||||
() => logState.buffer.slice(logState.visibleFrom),
|
() => logState.buffer.slice(logState.visibleFrom),
|
||||||
[logState.buffer, logState.visibleFrom]
|
[logState.buffer, logState.visibleFrom]
|
||||||
);
|
);
|
||||||
|
|
||||||
|
const trimmedSearchQuery = deferredSearchQuery.trim();
|
||||||
|
const isSearching = trimmedSearchQuery.length > 0;
|
||||||
|
const baseLines = isSearching ? logState.buffer : visibleLines;
|
||||||
|
|
||||||
|
const { filteredLines, removedCount } = useMemo(() => {
|
||||||
|
let working = baseLines;
|
||||||
|
let removed = 0;
|
||||||
|
|
||||||
|
if (hideManagementLogs) {
|
||||||
|
const next: string[] = [];
|
||||||
|
for (const line of working) {
|
||||||
|
if (line.includes(MANAGEMENT_API_PREFIX)) {
|
||||||
|
removed += 1;
|
||||||
|
} else {
|
||||||
|
next.push(line);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
working = next;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (trimmedSearchQuery) {
|
||||||
|
const queryLowered = trimmedSearchQuery.toLowerCase();
|
||||||
|
const next: string[] = [];
|
||||||
|
for (const line of working) {
|
||||||
|
if (line.toLowerCase().includes(queryLowered)) {
|
||||||
|
next.push(line);
|
||||||
|
} else {
|
||||||
|
removed += 1;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
working = next;
|
||||||
|
}
|
||||||
|
|
||||||
|
return { filteredLines: working, removedCount: removed };
|
||||||
|
}, [baseLines, hideManagementLogs, trimmedSearchQuery]);
|
||||||
|
|
||||||
const parsedVisibleLines = useMemo(
|
const parsedVisibleLines = useMemo(
|
||||||
() => visibleLines.map((line) => parseLogLine(line)),
|
() => filteredLines.map((line) => parseLogLine(line)),
|
||||||
[visibleLines]
|
[filteredLines]
|
||||||
);
|
);
|
||||||
const canLoadMore = logState.visibleFrom > 0;
|
|
||||||
|
const canLoadMore = !isSearching && logState.visibleFrom > 0;
|
||||||
|
|
||||||
const handleLogScroll = () => {
|
const handleLogScroll = () => {
|
||||||
const node = logViewerRef.current;
|
const node = logViewerRef.current;
|
||||||
if (!node) return;
|
if (!node) return;
|
||||||
|
if (isSearching) return;
|
||||||
if (!canLoadMore) return;
|
if (!canLoadMore) return;
|
||||||
if (pendingPrependScrollRef.current) return;
|
if (pendingPrependScrollRef.current) return;
|
||||||
if (node.scrollTop > LOAD_MORE_THRESHOLD_PX) return;
|
if (node.scrollTop > LOAD_MORE_THRESHOLD_PX) return;
|
||||||
|
|
||||||
pendingPrependScrollRef.current = { scrollHeight: node.scrollHeight, scrollTop: node.scrollTop };
|
pendingPrependScrollRef.current = {
|
||||||
|
scrollHeight: node.scrollHeight,
|
||||||
|
scrollTop: node.scrollTop,
|
||||||
|
};
|
||||||
setLogState((prev) => ({
|
setLogState((prev) => ({
|
||||||
...prev,
|
...prev,
|
||||||
visibleFrom: Math.max(prev.visibleFrom - LOAD_MORE_LINES, 0)
|
visibleFrom: Math.max(prev.visibleFrom - LOAD_MORE_LINES, 0),
|
||||||
}));
|
}));
|
||||||
};
|
};
|
||||||
|
|
||||||
@@ -464,185 +587,264 @@ export function LogsPage() {
|
|||||||
pendingPrependScrollRef.current = null;
|
pendingPrependScrollRef.current = null;
|
||||||
}, [logState.visibleFrom]);
|
}, [logState.visibleFrom]);
|
||||||
|
|
||||||
|
const copyLogLine = async (raw: string) => {
|
||||||
|
const ok = await copyToClipboard(raw);
|
||||||
|
if (ok) {
|
||||||
|
showNotification(t('logs.copy_success', { defaultValue: 'Copied to clipboard' }), 'success');
|
||||||
|
} else {
|
||||||
|
showNotification(t('logs.copy_failed', { defaultValue: 'Copy failed' }), 'error');
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div className={styles.container}>
|
<div className={styles.container}>
|
||||||
<h1 className={styles.pageTitle}>{t('logs.title')}</h1>
|
<h1 className={styles.pageTitle}>{t('logs.title')}</h1>
|
||||||
<div className={styles.content}>
|
<div className={styles.content}>
|
||||||
<Card
|
<Card
|
||||||
title={t('logs.log_content')}
|
title={t('logs.log_content')}
|
||||||
extra={
|
extra={
|
||||||
<div className={styles.toolbar}>
|
<div className={styles.toolbar}>
|
||||||
<Button
|
<Button
|
||||||
variant="secondary"
|
variant="secondary"
|
||||||
size="sm"
|
size="sm"
|
||||||
onClick={() => loadLogs(false)}
|
onClick={() => loadLogs(false)}
|
||||||
disabled={disableControls || loading}
|
disabled={disableControls || loading}
|
||||||
className={styles.actionButton}
|
className={styles.actionButton}
|
||||||
>
|
>
|
||||||
<span className={styles.buttonContent}>
|
<span className={styles.buttonContent}>
|
||||||
<IconRefreshCw size={16} />
|
<IconRefreshCw size={16} />
|
||||||
{t('logs.refresh_button')}
|
{t('logs.refresh_button')}
|
||||||
</span>
|
</span>
|
||||||
</Button>
|
</Button>
|
||||||
|
<ToggleSwitch
|
||||||
|
checked={autoRefresh}
|
||||||
|
onChange={(value) => setAutoRefresh(value)}
|
||||||
|
disabled={disableControls}
|
||||||
|
label={
|
||||||
|
<span className={styles.switchLabel}>
|
||||||
|
<IconTimer size={16} />
|
||||||
|
{t('logs.auto_refresh')}
|
||||||
|
</span>
|
||||||
|
}
|
||||||
|
/>
|
||||||
|
<Button
|
||||||
|
variant="secondary"
|
||||||
|
size="sm"
|
||||||
|
onClick={downloadLogs}
|
||||||
|
disabled={logState.buffer.length === 0}
|
||||||
|
className={styles.actionButton}
|
||||||
|
>
|
||||||
|
<span className={styles.buttonContent}>
|
||||||
|
<IconDownload size={16} />
|
||||||
|
{t('logs.download_button')}
|
||||||
|
</span>
|
||||||
|
</Button>
|
||||||
|
<Button
|
||||||
|
variant="danger"
|
||||||
|
size="sm"
|
||||||
|
onClick={clearLogs}
|
||||||
|
disabled={disableControls}
|
||||||
|
className={styles.actionButton}
|
||||||
|
>
|
||||||
|
<span className={styles.buttonContent}>
|
||||||
|
<IconTrash2 size={16} />
|
||||||
|
{t('logs.clear_button')}
|
||||||
|
</span>
|
||||||
|
</Button>
|
||||||
|
</div>
|
||||||
|
}
|
||||||
|
>
|
||||||
|
{error && <div className="error-box">{error}</div>}
|
||||||
|
|
||||||
|
<div className={styles.filters}>
|
||||||
|
<div className={styles.searchWrapper}>
|
||||||
|
<Input
|
||||||
|
value={searchQuery}
|
||||||
|
onChange={(e) => setSearchQuery(e.target.value)}
|
||||||
|
placeholder={t('logs.search_placeholder')}
|
||||||
|
className={styles.searchInput}
|
||||||
|
rightElement={
|
||||||
|
searchQuery ? (
|
||||||
|
<button
|
||||||
|
type="button"
|
||||||
|
className={styles.searchClear}
|
||||||
|
onClick={() => setSearchQuery('')}
|
||||||
|
title="Clear"
|
||||||
|
aria-label="Clear"
|
||||||
|
>
|
||||||
|
<IconX size={16} />
|
||||||
|
</button>
|
||||||
|
) : (
|
||||||
|
<IconSearch size={16} className={styles.searchIcon} />
|
||||||
|
)
|
||||||
|
}
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
|
||||||
<ToggleSwitch
|
<ToggleSwitch
|
||||||
checked={autoRefresh}
|
checked={hideManagementLogs}
|
||||||
onChange={(value) => setAutoRefresh(value)}
|
onChange={setHideManagementLogs}
|
||||||
disabled={disableControls}
|
|
||||||
label={
|
label={
|
||||||
<span className={styles.switchLabel}>
|
<span className={styles.switchLabel}>
|
||||||
<IconTimer size={16} />
|
<IconEyeOff size={16} />
|
||||||
{t('logs.auto_refresh')}
|
{t('logs.hide_management_logs', { prefix: MANAGEMENT_API_PREFIX })}
|
||||||
</span>
|
</span>
|
||||||
}
|
}
|
||||||
/>
|
/>
|
||||||
<Button
|
|
||||||
variant="secondary"
|
<div className={styles.filterStats}>
|
||||||
size="sm"
|
<span>
|
||||||
onClick={downloadLogs}
|
{parsedVisibleLines.length} {t('logs.lines')}
|
||||||
disabled={logState.buffer.length === 0}
|
|
||||||
className={styles.actionButton}
|
|
||||||
>
|
|
||||||
<span className={styles.buttonContent}>
|
|
||||||
<IconDownload size={16} />
|
|
||||||
{t('logs.download_button')}
|
|
||||||
</span>
|
</span>
|
||||||
</Button>
|
{removedCount > 0 && (
|
||||||
<Button
|
<span className={styles.removedCount}>
|
||||||
variant="danger"
|
{t('logs.removed')} {removedCount}
|
||||||
size="sm"
|
|
||||||
onClick={clearLogs}
|
|
||||||
disabled={disableControls}
|
|
||||||
className={styles.actionButton}
|
|
||||||
>
|
|
||||||
<span className={styles.buttonContent}>
|
|
||||||
<IconTrash2 size={16} />
|
|
||||||
{t('logs.clear_button')}
|
|
||||||
</span>
|
|
||||||
</Button>
|
|
||||||
</div>
|
|
||||||
}
|
|
||||||
>
|
|
||||||
{error && <div className="error-box">{error}</div>}
|
|
||||||
{loading ? (
|
|
||||||
<div className="hint">{t('logs.loading')}</div>
|
|
||||||
) : logState.buffer.length > 0 ? (
|
|
||||||
<div ref={logViewerRef} className={styles.logPanel} onScroll={handleLogScroll}>
|
|
||||||
{canLoadMore && (
|
|
||||||
<div className={styles.loadMoreBanner}>
|
|
||||||
<span>{t('logs.load_more_hint')}</span>
|
|
||||||
<span className={styles.loadMoreCount}>
|
|
||||||
{t('logs.hidden_lines', { count: logState.visibleFrom })}
|
|
||||||
</span>
|
</span>
|
||||||
</div>
|
)}
|
||||||
)}
|
|
||||||
<div className={styles.logList}>
|
|
||||||
{parsedVisibleLines.map((line, index) => {
|
|
||||||
const rowClassNames = [styles.logRow];
|
|
||||||
if (line.level === 'warn') rowClassNames.push(styles.rowWarn);
|
|
||||||
if (line.level === 'error' || line.level === 'fatal') rowClassNames.push(styles.rowError);
|
|
||||||
return (
|
|
||||||
<div key={`${logState.visibleFrom + index}-${line.raw}`} className={rowClassNames.join(' ')}>
|
|
||||||
<div className={styles.timestamp}>{line.timestamp || ''}</div>
|
|
||||||
<div className={styles.rowMain}>
|
|
||||||
<div className={styles.rowMeta}>
|
|
||||||
{line.level && (
|
|
||||||
<span
|
|
||||||
className={[
|
|
||||||
styles.badge,
|
|
||||||
line.level === 'info' ? styles.levelInfo : '',
|
|
||||||
line.level === 'warn' ? styles.levelWarn : '',
|
|
||||||
line.level === 'error' || line.level === 'fatal' ? styles.levelError : '',
|
|
||||||
line.level === 'debug' ? styles.levelDebug : '',
|
|
||||||
line.level === 'trace' ? styles.levelTrace : ''
|
|
||||||
]
|
|
||||||
.filter(Boolean)
|
|
||||||
.join(' ')}
|
|
||||||
>
|
|
||||||
{line.level.toUpperCase()}
|
|
||||||
</span>
|
|
||||||
)}
|
|
||||||
|
|
||||||
{line.source && (
|
|
||||||
<span className={styles.source} title={line.source}>
|
|
||||||
{line.source}
|
|
||||||
</span>
|
|
||||||
)}
|
|
||||||
|
|
||||||
{typeof line.statusCode === 'number' && (
|
|
||||||
<span
|
|
||||||
className={[
|
|
||||||
styles.badge,
|
|
||||||
styles.statusBadge,
|
|
||||||
line.statusCode >= 200 && line.statusCode < 300
|
|
||||||
? styles.statusSuccess
|
|
||||||
: line.statusCode >= 300 && line.statusCode < 400
|
|
||||||
? styles.statusInfo
|
|
||||||
: line.statusCode >= 400 && line.statusCode < 500
|
|
||||||
? styles.statusWarn
|
|
||||||
: styles.statusError
|
|
||||||
].join(' ')}
|
|
||||||
>
|
|
||||||
{line.statusCode}
|
|
||||||
</span>
|
|
||||||
)}
|
|
||||||
|
|
||||||
{line.latency && <span className={styles.pill}>{line.latency}</span>}
|
|
||||||
{line.ip && <span className={styles.pill}>{line.ip}</span>}
|
|
||||||
|
|
||||||
{line.method && (
|
|
||||||
<span className={[styles.badge, styles.methodBadge].join(' ')}>
|
|
||||||
{line.method}
|
|
||||||
</span>
|
|
||||||
)}
|
|
||||||
{line.path && (
|
|
||||||
<span className={styles.path} title={line.path}>
|
|
||||||
{line.path}
|
|
||||||
</span>
|
|
||||||
)}
|
|
||||||
</div>
|
|
||||||
{line.message && <div className={styles.message}>{line.message}</div>}
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
);
|
|
||||||
})}
|
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
) : (
|
|
||||||
<EmptyState title={t('logs.empty_title')} description={t('logs.empty_desc')} />
|
|
||||||
)}
|
|
||||||
</Card>
|
|
||||||
|
|
||||||
<Card
|
{loading ? (
|
||||||
title={t('logs.error_logs_modal_title')}
|
<div className="hint">{t('logs.loading')}</div>
|
||||||
extra={
|
) : logState.buffer.length > 0 && parsedVisibleLines.length > 0 ? (
|
||||||
<Button variant="secondary" size="sm" onClick={loadErrorLogs} loading={loadingErrors}>
|
<div ref={logViewerRef} className={styles.logPanel} onScroll={handleLogScroll}>
|
||||||
{t('common.refresh')}
|
{canLoadMore && (
|
||||||
</Button>
|
<div className={styles.loadMoreBanner}>
|
||||||
}
|
<span>{t('logs.load_more_hint')}</span>
|
||||||
>
|
<span className={styles.loadMoreCount}>
|
||||||
{errorLogs.length === 0 ? (
|
{t('logs.hidden_lines', { count: logState.visibleFrom })}
|
||||||
<div className="hint">{t('logs.error_logs_empty')}</div>
|
</span>
|
||||||
) : (
|
</div>
|
||||||
<div className="item-list">
|
)}
|
||||||
{errorLogs.map((item) => (
|
<div className={styles.logList}>
|
||||||
<div key={item.name} className="item-row">
|
{parsedVisibleLines.map((line, index) => {
|
||||||
<div className="item-meta">
|
const rowClassNames = [styles.logRow];
|
||||||
<div className="item-title">{item.name}</div>
|
if (line.level === 'warn') rowClassNames.push(styles.rowWarn);
|
||||||
<div className="item-subtitle">
|
if (line.level === 'error' || line.level === 'fatal')
|
||||||
{item.size ? `${(item.size / 1024).toFixed(1)} KB` : ''}{' '}
|
rowClassNames.push(styles.rowError);
|
||||||
{item.modified ? formatUnixTimestamp(item.modified) : ''}
|
return (
|
||||||
|
<div
|
||||||
|
key={`${logState.visibleFrom + index}-${line.raw}`}
|
||||||
|
className={rowClassNames.join(' ')}
|
||||||
|
onDoubleClick={() => {
|
||||||
|
void copyLogLine(line.raw);
|
||||||
|
}}
|
||||||
|
title={t('logs.double_click_copy_hint', {
|
||||||
|
defaultValue: 'Double-click to copy',
|
||||||
|
})}
|
||||||
|
>
|
||||||
|
<div className={styles.timestamp}>{line.timestamp || ''}</div>
|
||||||
|
<div className={styles.rowMain}>
|
||||||
|
<div className={styles.rowMeta}>
|
||||||
|
{line.level && (
|
||||||
|
<span
|
||||||
|
className={[
|
||||||
|
styles.badge,
|
||||||
|
line.level === 'info' ? styles.levelInfo : '',
|
||||||
|
line.level === 'warn' ? styles.levelWarn : '',
|
||||||
|
line.level === 'error' || line.level === 'fatal'
|
||||||
|
? styles.levelError
|
||||||
|
: '',
|
||||||
|
line.level === 'debug' ? styles.levelDebug : '',
|
||||||
|
line.level === 'trace' ? styles.levelTrace : '',
|
||||||
|
]
|
||||||
|
.filter(Boolean)
|
||||||
|
.join(' ')}
|
||||||
|
>
|
||||||
|
{line.level.toUpperCase()}
|
||||||
|
</span>
|
||||||
|
)}
|
||||||
|
|
||||||
|
{line.source && (
|
||||||
|
<span className={styles.source} title={line.source}>
|
||||||
|
{line.source}
|
||||||
|
</span>
|
||||||
|
)}
|
||||||
|
|
||||||
|
{typeof line.statusCode === 'number' && (
|
||||||
|
<span
|
||||||
|
className={[
|
||||||
|
styles.badge,
|
||||||
|
styles.statusBadge,
|
||||||
|
line.statusCode >= 200 && line.statusCode < 300
|
||||||
|
? styles.statusSuccess
|
||||||
|
: line.statusCode >= 300 && line.statusCode < 400
|
||||||
|
? styles.statusInfo
|
||||||
|
: line.statusCode >= 400 && line.statusCode < 500
|
||||||
|
? styles.statusWarn
|
||||||
|
: styles.statusError,
|
||||||
|
].join(' ')}
|
||||||
|
>
|
||||||
|
{line.statusCode}
|
||||||
|
</span>
|
||||||
|
)}
|
||||||
|
|
||||||
|
{line.latency && <span className={styles.pill}>{line.latency}</span>}
|
||||||
|
{line.ip && <span className={styles.pill}>{line.ip}</span>}
|
||||||
|
|
||||||
|
{line.method && (
|
||||||
|
<span className={[styles.badge, styles.methodBadge].join(' ')}>
|
||||||
|
{line.method}
|
||||||
|
</span>
|
||||||
|
)}
|
||||||
|
{line.path && (
|
||||||
|
<span className={styles.path} title={line.path}>
|
||||||
|
{line.path}
|
||||||
|
</span>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
{line.message && <div className={styles.message}>{line.message}</div>}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
})}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
) : logState.buffer.length > 0 ? (
|
||||||
|
<EmptyState
|
||||||
|
title={t('logs.search_empty_title')}
|
||||||
|
description={t('logs.search_empty_desc')}
|
||||||
|
/>
|
||||||
|
) : (
|
||||||
|
<EmptyState title={t('logs.empty_title')} description={t('logs.empty_desc')} />
|
||||||
|
)}
|
||||||
|
</Card>
|
||||||
|
|
||||||
|
<Card
|
||||||
|
title={t('logs.error_logs_modal_title')}
|
||||||
|
extra={
|
||||||
|
<Button variant="secondary" size="sm" onClick={loadErrorLogs} loading={loadingErrors}>
|
||||||
|
{t('common.refresh')}
|
||||||
|
</Button>
|
||||||
|
}
|
||||||
|
>
|
||||||
|
{errorLogs.length === 0 ? (
|
||||||
|
<div className="hint">{t('logs.error_logs_empty')}</div>
|
||||||
|
) : (
|
||||||
|
<div className="item-list">
|
||||||
|
{errorLogs.map((item) => (
|
||||||
|
<div key={item.name} className="item-row">
|
||||||
|
<div className="item-meta">
|
||||||
|
<div className="item-title">{item.name}</div>
|
||||||
|
<div className="item-subtitle">
|
||||||
|
{item.size ? `${(item.size / 1024).toFixed(1)} KB` : ''}{' '}
|
||||||
|
{item.modified ? formatUnixTimestamp(item.modified) : ''}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div className="item-actions">
|
||||||
|
<Button
|
||||||
|
variant="secondary"
|
||||||
|
size="sm"
|
||||||
|
onClick={() => downloadErrorLog(item.name)}
|
||||||
|
>
|
||||||
|
{t('logs.error_logs_download')}
|
||||||
|
</Button>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<div className="item-actions">
|
))}
|
||||||
<Button variant="secondary" size="sm" onClick={() => downloadErrorLog(item.name)}>
|
</div>
|
||||||
{t('logs.error_logs_download')}
|
)}
|
||||||
</Button>
|
</Card>
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
))}
|
|
||||||
</div>
|
|
||||||
)}
|
|
||||||
</Card>
|
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
);
|
);
|
||||||
|
|||||||
@@ -59,3 +59,47 @@
|
|||||||
color: #3b82f6;
|
color: #3b82f6;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.callbackSection {
|
||||||
|
margin-top: $spacing-md;
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
gap: $spacing-xs;
|
||||||
|
}
|
||||||
|
|
||||||
|
.callbackActions {
|
||||||
|
display: flex;
|
||||||
|
gap: $spacing-md;
|
||||||
|
}
|
||||||
|
|
||||||
|
.authUrlBox {
|
||||||
|
background: var(--bg-secondary);
|
||||||
|
border: 1px dashed var(--border-color);
|
||||||
|
border-radius: $radius-md;
|
||||||
|
padding: $spacing-md;
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
gap: $spacing-xs;
|
||||||
|
}
|
||||||
|
|
||||||
|
.authUrlLabel {
|
||||||
|
color: var(--text-secondary);
|
||||||
|
font-size: 14px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.authUrlValue {
|
||||||
|
font-weight: 700;
|
||||||
|
color: var(--text-primary);
|
||||||
|
word-break: break-all;
|
||||||
|
overflow-wrap: anywhere;
|
||||||
|
line-height: 1.5;
|
||||||
|
max-width: 100%;
|
||||||
|
}
|
||||||
|
|
||||||
|
.authUrlActions {
|
||||||
|
display: flex;
|
||||||
|
flex-wrap: wrap;
|
||||||
|
align-items: center;
|
||||||
|
gap: $spacing-sm;
|
||||||
|
margin-top: $spacing-sm;
|
||||||
|
}
|
||||||
|
|||||||
@@ -1,11 +1,10 @@
|
|||||||
import { useEffect, useRef, useState, useMemo } from 'react';
|
import { useEffect, useRef, useState } from 'react';
|
||||||
import { useTranslation } from 'react-i18next';
|
import { useTranslation } from 'react-i18next';
|
||||||
import { Card } from '@/components/ui/Card';
|
import { Card } from '@/components/ui/Card';
|
||||||
import { Button } from '@/components/ui/Button';
|
import { Button } from '@/components/ui/Button';
|
||||||
import { Input } from '@/components/ui/Input';
|
import { Input } from '@/components/ui/Input';
|
||||||
import { useNotificationStore } from '@/stores';
|
import { useNotificationStore } from '@/stores';
|
||||||
import { oauthApi, type OAuthProvider, type IFlowCookieAuthResponse } from '@/services/api/oauth';
|
import { oauthApi, type OAuthProvider, type IFlowCookieAuthResponse } from '@/services/api/oauth';
|
||||||
import { isLocalhost } from '@/utils/connection';
|
|
||||||
import styles from './OAuthPage.module.scss';
|
import styles from './OAuthPage.module.scss';
|
||||||
|
|
||||||
interface ProviderState {
|
interface ProviderState {
|
||||||
@@ -14,6 +13,12 @@ interface ProviderState {
|
|||||||
status?: 'idle' | 'waiting' | 'success' | 'error';
|
status?: 'idle' | 'waiting' | 'success' | 'error';
|
||||||
error?: string;
|
error?: string;
|
||||||
polling?: boolean;
|
polling?: boolean;
|
||||||
|
projectId?: string;
|
||||||
|
projectIdError?: string;
|
||||||
|
callbackUrl?: string;
|
||||||
|
callbackSubmitting?: boolean;
|
||||||
|
callbackStatus?: 'success' | 'error';
|
||||||
|
callbackError?: string;
|
||||||
}
|
}
|
||||||
|
|
||||||
interface IFlowCookieState {
|
interface IFlowCookieState {
|
||||||
@@ -33,6 +38,8 @@ const PROVIDERS: { id: OAuthProvider; titleKey: string; hintKey: string; urlLabe
|
|||||||
{ id: 'iflow', titleKey: 'auth_login.iflow_oauth_title', hintKey: 'auth_login.iflow_oauth_hint', urlLabelKey: 'auth_login.iflow_oauth_url_label' }
|
{ id: 'iflow', titleKey: 'auth_login.iflow_oauth_title', hintKey: 'auth_login.iflow_oauth_hint', urlLabelKey: 'auth_login.iflow_oauth_url_label' }
|
||||||
];
|
];
|
||||||
|
|
||||||
|
const CALLBACK_SUPPORTED: OAuthProvider[] = ['codex', 'anthropic', 'antigravity', 'gemini-cli', 'iflow'];
|
||||||
|
|
||||||
export function OAuthPage() {
|
export function OAuthPage() {
|
||||||
const { t } = useTranslation();
|
const { t } = useTranslation();
|
||||||
const { showNotification } = useNotificationStore();
|
const { showNotification } = useNotificationStore();
|
||||||
@@ -40,15 +47,19 @@ export function OAuthPage() {
|
|||||||
const [iflowCookie, setIflowCookie] = useState<IFlowCookieState>({ cookie: '', loading: false });
|
const [iflowCookie, setIflowCookie] = useState<IFlowCookieState>({ cookie: '', loading: false });
|
||||||
const timers = useRef<Record<string, number>>({});
|
const timers = useRef<Record<string, number>>({});
|
||||||
|
|
||||||
// 检测是否为本地访问
|
|
||||||
const isLocal = useMemo(() => isLocalhost(window.location.hostname), []);
|
|
||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
return () => {
|
return () => {
|
||||||
Object.values(timers.current).forEach((timer) => window.clearInterval(timer));
|
Object.values(timers.current).forEach((timer) => window.clearInterval(timer));
|
||||||
};
|
};
|
||||||
}, []);
|
}, []);
|
||||||
|
|
||||||
|
const updateProviderState = (provider: OAuthProvider, next: Partial<ProviderState>) => {
|
||||||
|
setStates((prev) => ({
|
||||||
|
...prev,
|
||||||
|
[provider]: { ...(prev[provider] ?? {}), ...next }
|
||||||
|
}));
|
||||||
|
};
|
||||||
|
|
||||||
const startPolling = (provider: OAuthProvider, state: string) => {
|
const startPolling = (provider: OAuthProvider, state: string) => {
|
||||||
if (timers.current[provider]) {
|
if (timers.current[provider]) {
|
||||||
clearInterval(timers.current[provider]);
|
clearInterval(timers.current[provider]);
|
||||||
@@ -57,27 +68,18 @@ export function OAuthPage() {
|
|||||||
try {
|
try {
|
||||||
const res = await oauthApi.getAuthStatus(state);
|
const res = await oauthApi.getAuthStatus(state);
|
||||||
if (res.status === 'ok') {
|
if (res.status === 'ok') {
|
||||||
setStates((prev) => ({
|
updateProviderState(provider, { status: 'success', polling: false });
|
||||||
...prev,
|
|
||||||
[provider]: { ...prev[provider], status: 'success', polling: false }
|
|
||||||
}));
|
|
||||||
showNotification(t('auth_login.codex_oauth_status_success'), 'success');
|
showNotification(t('auth_login.codex_oauth_status_success'), 'success');
|
||||||
window.clearInterval(timer);
|
window.clearInterval(timer);
|
||||||
delete timers.current[provider];
|
delete timers.current[provider];
|
||||||
} else if (res.status === 'error') {
|
} else if (res.status === 'error') {
|
||||||
setStates((prev) => ({
|
updateProviderState(provider, { status: 'error', error: res.error, polling: false });
|
||||||
...prev,
|
|
||||||
[provider]: { ...prev[provider], status: 'error', error: res.error, polling: false }
|
|
||||||
}));
|
|
||||||
showNotification(`${t('auth_login.codex_oauth_status_error')} ${res.error || ''}`, 'error');
|
showNotification(`${t('auth_login.codex_oauth_status_error')} ${res.error || ''}`, 'error');
|
||||||
window.clearInterval(timer);
|
window.clearInterval(timer);
|
||||||
delete timers.current[provider];
|
delete timers.current[provider];
|
||||||
}
|
}
|
||||||
} catch (err: any) {
|
} catch (err: any) {
|
||||||
setStates((prev) => ({
|
updateProviderState(provider, { status: 'error', error: err?.message, polling: false });
|
||||||
...prev,
|
|
||||||
[provider]: { ...prev[provider], status: 'error', error: err?.message, polling: false }
|
|
||||||
}));
|
|
||||||
window.clearInterval(timer);
|
window.clearInterval(timer);
|
||||||
delete timers.current[provider];
|
delete timers.current[provider];
|
||||||
}
|
}
|
||||||
@@ -86,24 +88,35 @@ export function OAuthPage() {
|
|||||||
};
|
};
|
||||||
|
|
||||||
const startAuth = async (provider: OAuthProvider) => {
|
const startAuth = async (provider: OAuthProvider) => {
|
||||||
setStates((prev) => ({
|
const projectId = provider === 'gemini-cli' ? (states[provider]?.projectId || '').trim() : undefined;
|
||||||
...prev,
|
if (provider === 'gemini-cli' && !projectId) {
|
||||||
[provider]: { ...prev[provider], status: 'waiting', polling: true, error: undefined }
|
const message = t('auth_login.gemini_cli_project_id_required');
|
||||||
}));
|
updateProviderState(provider, { projectIdError: message });
|
||||||
|
showNotification(message, 'warning');
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
if (provider === 'gemini-cli') {
|
||||||
|
updateProviderState(provider, { projectIdError: undefined });
|
||||||
|
}
|
||||||
|
updateProviderState(provider, {
|
||||||
|
status: 'waiting',
|
||||||
|
polling: true,
|
||||||
|
error: undefined,
|
||||||
|
callbackStatus: undefined,
|
||||||
|
callbackError: undefined,
|
||||||
|
callbackUrl: ''
|
||||||
|
});
|
||||||
try {
|
try {
|
||||||
const res = await oauthApi.startAuth(provider);
|
const res = await oauthApi.startAuth(
|
||||||
setStates((prev) => ({
|
provider,
|
||||||
...prev,
|
provider === 'gemini-cli' ? { projectId: projectId! } : undefined
|
||||||
[provider]: { ...prev[provider], url: res.url, state: res.state, status: 'waiting', polling: true }
|
);
|
||||||
}));
|
updateProviderState(provider, { url: res.url, state: res.state, status: 'waiting', polling: true });
|
||||||
if (res.state) {
|
if (res.state) {
|
||||||
startPolling(provider, res.state);
|
startPolling(provider, res.state);
|
||||||
}
|
}
|
||||||
} catch (err: any) {
|
} catch (err: any) {
|
||||||
setStates((prev) => ({
|
updateProviderState(provider, { status: 'error', error: err?.message, polling: false });
|
||||||
...prev,
|
|
||||||
[provider]: { ...prev[provider], status: 'error', error: err?.message, polling: false }
|
|
||||||
}));
|
|
||||||
showNotification(`${t('auth_login.codex_oauth_start_error')} ${err?.message || ''}`, 'error');
|
showNotification(`${t('auth_login.codex_oauth_start_error')} ${err?.message || ''}`, 'error');
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
@@ -118,6 +131,40 @@ export function OAuthPage() {
|
|||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
const submitCallback = async (provider: OAuthProvider) => {
|
||||||
|
const redirectUrl = (states[provider]?.callbackUrl || '').trim();
|
||||||
|
if (!redirectUrl) {
|
||||||
|
showNotification(t('auth_login.oauth_callback_required'), 'warning');
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
updateProviderState(provider, {
|
||||||
|
callbackSubmitting: true,
|
||||||
|
callbackStatus: undefined,
|
||||||
|
callbackError: undefined
|
||||||
|
});
|
||||||
|
try {
|
||||||
|
await oauthApi.submitCallback(provider, redirectUrl);
|
||||||
|
updateProviderState(provider, { callbackSubmitting: false, callbackStatus: 'success' });
|
||||||
|
showNotification(t('auth_login.oauth_callback_success'), 'success');
|
||||||
|
} catch (err: any) {
|
||||||
|
const errorMessage =
|
||||||
|
err?.status === 404
|
||||||
|
? t('auth_login.oauth_callback_upgrade_hint', {
|
||||||
|
defaultValue: 'Please update CLI Proxy API or check the connection.'
|
||||||
|
})
|
||||||
|
: err?.message;
|
||||||
|
updateProviderState(provider, {
|
||||||
|
callbackSubmitting: false,
|
||||||
|
callbackStatus: 'error',
|
||||||
|
callbackError: errorMessage
|
||||||
|
});
|
||||||
|
const notificationMessage = errorMessage
|
||||||
|
? `${t('auth_login.oauth_callback_error')} ${errorMessage}`
|
||||||
|
: t('auth_login.oauth_callback_error');
|
||||||
|
showNotification(notificationMessage, 'error');
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
const submitIflowCookie = async () => {
|
const submitIflowCookie = async () => {
|
||||||
const cookie = iflowCookie.cookie.trim();
|
const cookie = iflowCookie.cookie.trim();
|
||||||
if (!cookie) {
|
if (!cookie) {
|
||||||
@@ -164,36 +211,38 @@ export function OAuthPage() {
|
|||||||
<div className={styles.content}>
|
<div className={styles.content}>
|
||||||
{PROVIDERS.map((provider) => {
|
{PROVIDERS.map((provider) => {
|
||||||
const state = states[provider.id] || {};
|
const state = states[provider.id] || {};
|
||||||
// 非本地访问时禁用所有 OAuth 登录方式
|
const canSubmitCallback = CALLBACK_SUPPORTED.includes(provider.id) && Boolean(state.url);
|
||||||
const isDisabled = !isLocal;
|
|
||||||
return (
|
return (
|
||||||
<div
|
<div key={provider.id}>
|
||||||
key={provider.id}
|
|
||||||
style={isDisabled ? { opacity: 0.6, pointerEvents: 'none' } : undefined}
|
|
||||||
>
|
|
||||||
<Card
|
<Card
|
||||||
title={t(provider.titleKey)}
|
title={t(provider.titleKey)}
|
||||||
extra={
|
extra={
|
||||||
<Button
|
<Button onClick={() => startAuth(provider.id)} loading={state.polling}>
|
||||||
onClick={() => startAuth(provider.id)}
|
|
||||||
loading={state.polling}
|
|
||||||
disabled={isDisabled}
|
|
||||||
>
|
|
||||||
{t('common.login')}
|
{t('common.login')}
|
||||||
</Button>
|
</Button>
|
||||||
}
|
}
|
||||||
>
|
>
|
||||||
<div className="hint">{t(provider.hintKey)}</div>
|
<div className="hint">{t(provider.hintKey)}</div>
|
||||||
{isDisabled && (
|
{provider.id === 'gemini-cli' && (
|
||||||
<div className="status-badge warning" style={{ marginTop: 8 }}>
|
<Input
|
||||||
{t('auth_login.remote_access_disabled')}
|
label={t('auth_login.gemini_cli_project_id_label')}
|
||||||
</div>
|
hint={t('auth_login.gemini_cli_project_id_hint')}
|
||||||
|
value={state.projectId || ''}
|
||||||
|
error={state.projectIdError}
|
||||||
|
onChange={(e) =>
|
||||||
|
updateProviderState(provider.id, {
|
||||||
|
projectId: e.target.value,
|
||||||
|
projectIdError: undefined
|
||||||
|
})
|
||||||
|
}
|
||||||
|
placeholder={t('auth_login.gemini_cli_project_id_placeholder')}
|
||||||
|
/>
|
||||||
)}
|
)}
|
||||||
{!isDisabled && state.url && (
|
{state.url && (
|
||||||
<div className="connection-box">
|
<div className={`connection-box ${styles.authUrlBox}`}>
|
||||||
<div className="label">{t(provider.urlLabelKey)}</div>
|
<div className={styles.authUrlLabel}>{t(provider.urlLabelKey)}</div>
|
||||||
<div className="value">{state.url}</div>
|
<div className={styles.authUrlValue}>{state.url}</div>
|
||||||
<div className="item-actions" style={{ marginTop: 8 }}>
|
<div className={styles.authUrlActions}>
|
||||||
<Button variant="secondary" size="sm" onClick={() => copyLink(state.url!)}>
|
<Button variant="secondary" size="sm" onClick={() => copyLink(state.url!)}>
|
||||||
{t('auth_login.codex_copy_link')}
|
{t('auth_login.codex_copy_link')}
|
||||||
</Button>
|
</Button>
|
||||||
@@ -207,7 +256,44 @@ export function OAuthPage() {
|
|||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
)}
|
)}
|
||||||
{!isDisabled && state.status && state.status !== 'idle' && (
|
{canSubmitCallback && (
|
||||||
|
<div className={styles.callbackSection}>
|
||||||
|
<Input
|
||||||
|
label={t('auth_login.oauth_callback_label')}
|
||||||
|
hint={t('auth_login.oauth_callback_hint')}
|
||||||
|
value={state.callbackUrl || ''}
|
||||||
|
onChange={(e) =>
|
||||||
|
updateProviderState(provider.id, {
|
||||||
|
callbackUrl: e.target.value,
|
||||||
|
callbackStatus: undefined,
|
||||||
|
callbackError: undefined
|
||||||
|
})
|
||||||
|
}
|
||||||
|
placeholder={t('auth_login.oauth_callback_placeholder')}
|
||||||
|
/>
|
||||||
|
<div className={styles.callbackActions}>
|
||||||
|
<Button
|
||||||
|
variant="secondary"
|
||||||
|
size="sm"
|
||||||
|
onClick={() => submitCallback(provider.id)}
|
||||||
|
loading={state.callbackSubmitting}
|
||||||
|
>
|
||||||
|
{t('auth_login.oauth_callback_button')}
|
||||||
|
</Button>
|
||||||
|
</div>
|
||||||
|
{state.callbackStatus === 'success' && state.status === 'waiting' && (
|
||||||
|
<div className="status-badge success" style={{ marginTop: 8 }}>
|
||||||
|
{t('auth_login.oauth_callback_status_success')}
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
{state.callbackStatus === 'error' && (
|
||||||
|
<div className="status-badge error" style={{ marginTop: 8 }}>
|
||||||
|
{t('auth_login.oauth_callback_status_error')} {state.callbackError || ''}
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
{state.status && state.status !== 'idle' && (
|
||||||
<div className="status-badge" style={{ marginTop: 8 }}>
|
<div className="status-badge" style={{ marginTop: 8 }}>
|
||||||
{state.status === 'success'
|
{state.status === 'success'
|
||||||
? t('auth_login.codex_oauth_status_success')
|
? t('auth_login.codex_oauth_status_success')
|
||||||
|
|||||||
@@ -3,9 +3,11 @@
|
|||||||
|
|
||||||
.container {
|
.container {
|
||||||
width: 100%;
|
width: 100%;
|
||||||
|
min-height: 100%;
|
||||||
display: flex;
|
display: flex;
|
||||||
flex-direction: column;
|
flex-direction: column;
|
||||||
gap: 20px;
|
gap: 20px;
|
||||||
|
position: relative;
|
||||||
}
|
}
|
||||||
|
|
||||||
.header {
|
.header {
|
||||||
@@ -39,6 +41,45 @@
|
|||||||
padding: 16px;
|
padding: 16px;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.loadingOverlay {
|
||||||
|
position: absolute;
|
||||||
|
inset: 0;
|
||||||
|
z-index: 20;
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
justify-content: center;
|
||||||
|
background: rgba(243, 244, 246, 0.75);
|
||||||
|
backdrop-filter: blur(6px);
|
||||||
|
-webkit-backdrop-filter: blur(6px);
|
||||||
|
}
|
||||||
|
|
||||||
|
:global([data-theme='dark']) .loadingOverlay {
|
||||||
|
background: rgba(25, 25, 25, 0.72);
|
||||||
|
}
|
||||||
|
|
||||||
|
.loadingOverlayContent {
|
||||||
|
display: inline-flex;
|
||||||
|
align-items: center;
|
||||||
|
gap: 10px;
|
||||||
|
padding: 12px 16px;
|
||||||
|
border-radius: $radius-lg;
|
||||||
|
border: 1px solid var(--border-color);
|
||||||
|
background: var(--bg-primary);
|
||||||
|
box-shadow: var(--shadow-lg);
|
||||||
|
}
|
||||||
|
|
||||||
|
.loadingOverlaySpinner {
|
||||||
|
border-color: rgba(59, 130, 246, 0.25);
|
||||||
|
border-top-color: var(--primary-color);
|
||||||
|
box-shadow: 0 0 10px rgba(59, 130, 246, 0.25);
|
||||||
|
}
|
||||||
|
|
||||||
|
.loadingOverlayText {
|
||||||
|
font-size: 13px;
|
||||||
|
font-weight: 600;
|
||||||
|
color: var(--text-secondary);
|
||||||
|
}
|
||||||
|
|
||||||
// Stats Grid
|
// Stats Grid
|
||||||
.statsGrid {
|
.statsGrid {
|
||||||
display: grid;
|
display: grid;
|
||||||
|
|||||||
@@ -16,6 +16,7 @@ import { Line } from 'react-chartjs-2';
|
|||||||
import { Card } from '@/components/ui/Card';
|
import { Card } from '@/components/ui/Card';
|
||||||
import { Button } from '@/components/ui/Button';
|
import { Button } from '@/components/ui/Button';
|
||||||
import { Input } from '@/components/ui/Input';
|
import { Input } from '@/components/ui/Input';
|
||||||
|
import { LoadingSpinner } from '@/components/ui/LoadingSpinner';
|
||||||
import { IconDiamond, IconDollarSign, IconSatellite, IconTimer, IconTrendingUp } from '@/components/ui/icons';
|
import { IconDiamond, IconDollarSign, IconSatellite, IconTimer, IconTrendingUp } from '@/components/ui/icons';
|
||||||
import { useMediaQuery } from '@/hooks/useMediaQuery';
|
import { useMediaQuery } from '@/hooks/useMediaQuery';
|
||||||
import { useThemeStore } from '@/stores';
|
import { useThemeStore } from '@/stores';
|
||||||
@@ -516,6 +517,14 @@ export function UsagePage() {
|
|||||||
|
|
||||||
return (
|
return (
|
||||||
<div className={styles.container}>
|
<div className={styles.container}>
|
||||||
|
{loading && !usage && (
|
||||||
|
<div className={styles.loadingOverlay} aria-busy="true">
|
||||||
|
<div className={styles.loadingOverlayContent}>
|
||||||
|
<LoadingSpinner size={28} className={styles.loadingOverlaySpinner} />
|
||||||
|
<span className={styles.loadingOverlayText}>{t('common.loading')}</span>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
<div className={styles.header}>
|
<div className={styles.header}>
|
||||||
<h1 className={styles.pageTitle}>{t('usage_stats.title')}</h1>
|
<h1 className={styles.pageTitle}>{t('usage_stats.title')}</h1>
|
||||||
<Button
|
<Button
|
||||||
|
|||||||
@@ -18,9 +18,6 @@ export const ampcodeApi = {
|
|||||||
updateUpstreamApiKey: (apiKey: string) => apiClient.put('/ampcode/upstream-api-key', { value: apiKey }),
|
updateUpstreamApiKey: (apiKey: string) => apiClient.put('/ampcode/upstream-api-key', { value: apiKey }),
|
||||||
clearUpstreamApiKey: () => apiClient.delete('/ampcode/upstream-api-key'),
|
clearUpstreamApiKey: () => apiClient.delete('/ampcode/upstream-api-key'),
|
||||||
|
|
||||||
updateRestrictManagementToLocalhost: (enabled: boolean) =>
|
|
||||||
apiClient.put('/ampcode/restrict-management-to-localhost', { value: enabled }),
|
|
||||||
|
|
||||||
async getModelMappings(): Promise<AmpcodeModelMapping[]> {
|
async getModelMappings(): Promise<AmpcodeModelMapping[]> {
|
||||||
const data = await apiClient.get('/ampcode/model-mappings');
|
const data = await apiClient.get('/ampcode/model-mappings');
|
||||||
const list = data?.['model-mappings'] ?? data?.modelMappings ?? data?.items ?? data;
|
const list = data?.['model-mappings'] ?? data?.modelMappings ?? data?.items ?? data;
|
||||||
|
|||||||
@@ -82,6 +82,10 @@ class ApiClient {
|
|||||||
(config) => {
|
(config) => {
|
||||||
// 设置 baseURL
|
// 设置 baseURL
|
||||||
config.baseURL = this.apiBase;
|
config.baseURL = this.apiBase;
|
||||||
|
if (config.url) {
|
||||||
|
// Normalize deprecated Gemini endpoint to the current path.
|
||||||
|
config.url = config.url.replace(/\/generative-language-api-key\b/g, '/gemini-api-key');
|
||||||
|
}
|
||||||
|
|
||||||
// 添加认证头
|
// 添加认证头
|
||||||
if (this.managementKey) {
|
if (this.managementKey) {
|
||||||
|
|||||||
@@ -3,6 +3,7 @@
|
|||||||
*/
|
*/
|
||||||
|
|
||||||
import { apiClient } from './client';
|
import { apiClient } from './client';
|
||||||
|
import { LOGS_TIMEOUT_MS } from '@/utils/constants';
|
||||||
|
|
||||||
export interface LogsQuery {
|
export interface LogsQuery {
|
||||||
after?: number;
|
after?: number;
|
||||||
@@ -14,16 +15,28 @@ export interface LogsResponse {
|
|||||||
'latest-timestamp': number;
|
'latest-timestamp': number;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
export interface ErrorLogFile {
|
||||||
|
name: string;
|
||||||
|
size?: number;
|
||||||
|
modified?: number;
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface ErrorLogsResponse {
|
||||||
|
files?: ErrorLogFile[];
|
||||||
|
}
|
||||||
|
|
||||||
export const logsApi = {
|
export const logsApi = {
|
||||||
fetchLogs: (params: LogsQuery = {}): Promise<LogsResponse> =>
|
fetchLogs: (params: LogsQuery = {}): Promise<LogsResponse> =>
|
||||||
apiClient.get('/logs', { params }),
|
apiClient.get('/logs', { params, timeout: LOGS_TIMEOUT_MS }),
|
||||||
|
|
||||||
clearLogs: () => apiClient.delete('/logs'),
|
clearLogs: () => apiClient.delete('/logs'),
|
||||||
|
|
||||||
fetchErrorLogs: () => apiClient.get('/request-error-logs'),
|
fetchErrorLogs: (): Promise<ErrorLogsResponse> =>
|
||||||
|
apiClient.get('/request-error-logs', { timeout: LOGS_TIMEOUT_MS }),
|
||||||
|
|
||||||
downloadErrorLog: (filename: string) =>
|
downloadErrorLog: (filename: string) =>
|
||||||
apiClient.getRaw(`/request-error-logs/${encodeURIComponent(filename)}`, {
|
apiClient.getRaw(`/request-error-logs/${encodeURIComponent(filename)}`, {
|
||||||
responseType: 'blob'
|
responseType: 'blob',
|
||||||
})
|
timeout: LOGS_TIMEOUT_MS
|
||||||
|
}),
|
||||||
};
|
};
|
||||||
|
|||||||
@@ -17,6 +17,10 @@ export interface OAuthStartResponse {
|
|||||||
state?: string;
|
state?: string;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
export interface OAuthCallbackResponse {
|
||||||
|
status: 'ok';
|
||||||
|
}
|
||||||
|
|
||||||
export interface IFlowCookieAuthResponse {
|
export interface IFlowCookieAuthResponse {
|
||||||
status: 'ok' | 'error';
|
status: 'ok' | 'error';
|
||||||
error?: string;
|
error?: string;
|
||||||
@@ -27,18 +31,37 @@ export interface IFlowCookieAuthResponse {
|
|||||||
}
|
}
|
||||||
|
|
||||||
const WEBUI_SUPPORTED: OAuthProvider[] = ['codex', 'anthropic', 'antigravity', 'gemini-cli', 'iflow'];
|
const WEBUI_SUPPORTED: OAuthProvider[] = ['codex', 'anthropic', 'antigravity', 'gemini-cli', 'iflow'];
|
||||||
|
const CALLBACK_PROVIDER_MAP: Partial<Record<OAuthProvider, string>> = {
|
||||||
|
'gemini-cli': 'gemini'
|
||||||
|
};
|
||||||
|
|
||||||
export const oauthApi = {
|
export const oauthApi = {
|
||||||
startAuth: (provider: OAuthProvider) =>
|
startAuth: (provider: OAuthProvider, options?: { projectId?: string }) => {
|
||||||
apiClient.get<OAuthStartResponse>(`/${provider}-auth-url`, {
|
const params: Record<string, string | boolean> = {};
|
||||||
params: WEBUI_SUPPORTED.includes(provider) ? { is_webui: true } : undefined
|
if (WEBUI_SUPPORTED.includes(provider)) {
|
||||||
}),
|
params.is_webui = true;
|
||||||
|
}
|
||||||
|
if (provider === 'gemini-cli' && options?.projectId) {
|
||||||
|
params.project_id = options.projectId;
|
||||||
|
}
|
||||||
|
return apiClient.get<OAuthStartResponse>(`/${provider}-auth-url`, {
|
||||||
|
params: Object.keys(params).length ? params : undefined
|
||||||
|
});
|
||||||
|
},
|
||||||
|
|
||||||
getAuthStatus: (state: string) =>
|
getAuthStatus: (state: string) =>
|
||||||
apiClient.get<{ status: 'ok' | 'wait' | 'error'; error?: string }>(`/get-auth-status`, {
|
apiClient.get<{ status: 'ok' | 'wait' | 'error'; error?: string }>(`/get-auth-status`, {
|
||||||
params: { state }
|
params: { state }
|
||||||
}),
|
}),
|
||||||
|
|
||||||
|
submitCallback: (provider: OAuthProvider, redirectUrl: string) => {
|
||||||
|
const callbackProvider = CALLBACK_PROVIDER_MAP[provider] ?? provider;
|
||||||
|
return apiClient.post<OAuthCallbackResponse>('/oauth-callback', {
|
||||||
|
provider: callbackProvider,
|
||||||
|
redirect_url: redirectUrl
|
||||||
|
});
|
||||||
|
},
|
||||||
|
|
||||||
/** iFlow cookie 认证 */
|
/** iFlow cookie 认证 */
|
||||||
iflowCookieAuth: (cookie: string) =>
|
iflowCookieAuth: (cookie: string) =>
|
||||||
apiClient.post<IFlowCookieAuthResponse>('/iflow-auth-url', { cookie })
|
apiClient.post<IFlowCookieAuthResponse>('/iflow-auth-url', { cookie })
|
||||||
|
|||||||
@@ -205,15 +205,6 @@ const normalizeAmpcodeConfig = (payload: any): AmpcodeConfig | undefined => {
|
|||||||
const upstreamApiKey = source['upstream-api-key'] ?? source.upstreamApiKey ?? source['upstream_api_key'];
|
const upstreamApiKey = source['upstream-api-key'] ?? source.upstreamApiKey ?? source['upstream_api_key'];
|
||||||
if (upstreamApiKey) config.upstreamApiKey = String(upstreamApiKey);
|
if (upstreamApiKey) config.upstreamApiKey = String(upstreamApiKey);
|
||||||
|
|
||||||
const restrictManagementToLocalhost = normalizeBoolean(
|
|
||||||
source['restrict-management-to-localhost'] ??
|
|
||||||
source.restrictManagementToLocalhost ??
|
|
||||||
source['restrict_management_to_localhost']
|
|
||||||
);
|
|
||||||
if (restrictManagementToLocalhost !== undefined) {
|
|
||||||
config.restrictManagementToLocalhost = restrictManagementToLocalhost;
|
|
||||||
}
|
|
||||||
|
|
||||||
const forceModelMappings = normalizeBoolean(
|
const forceModelMappings = normalizeBoolean(
|
||||||
source['force-model-mappings'] ?? source.forceModelMappings ?? source['force_model_mappings']
|
source['force-model-mappings'] ?? source.forceModelMappings ?? source['force_model_mappings']
|
||||||
);
|
);
|
||||||
|
|||||||
@@ -5,11 +5,13 @@
|
|||||||
import { apiClient } from './client';
|
import { apiClient } from './client';
|
||||||
import { computeKeyStats, KeyStats } from '@/utils/usage';
|
import { computeKeyStats, KeyStats } from '@/utils/usage';
|
||||||
|
|
||||||
|
const USAGE_TIMEOUT_MS = 60 * 1000;
|
||||||
|
|
||||||
export const usageApi = {
|
export const usageApi = {
|
||||||
/**
|
/**
|
||||||
* 获取使用统计原始数据
|
* 获取使用统计原始数据
|
||||||
*/
|
*/
|
||||||
getUsage: () => apiClient.get('/usage'),
|
getUsage: () => apiClient.get('/usage', { timeout: USAGE_TIMEOUT_MS }),
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* 计算密钥成功/失败统计,必要时会先获取 usage 数据
|
* 计算密钥成功/失败统计,必要时会先获取 usage 数据
|
||||||
@@ -17,7 +19,7 @@ export const usageApi = {
|
|||||||
async getKeyStats(usageData?: any): Promise<KeyStats> {
|
async getKeyStats(usageData?: any): Promise<KeyStats> {
|
||||||
let payload = usageData;
|
let payload = usageData;
|
||||||
if (!payload) {
|
if (!payload) {
|
||||||
const response = await apiClient.get('/usage');
|
const response = await apiClient.get('/usage', { timeout: USAGE_TIMEOUT_MS });
|
||||||
payload = response?.usage ?? response;
|
payload = response?.usage ?? response;
|
||||||
}
|
}
|
||||||
return computeKeyStats(payload);
|
return computeKeyStats(payload);
|
||||||
|
|||||||
@@ -10,7 +10,6 @@ export interface AmpcodeModelMapping {
|
|||||||
export interface AmpcodeConfig {
|
export interface AmpcodeConfig {
|
||||||
upstreamUrl?: string;
|
upstreamUrl?: string;
|
||||||
upstreamApiKey?: string;
|
upstreamApiKey?: string;
|
||||||
restrictManagementToLocalhost?: boolean;
|
|
||||||
modelMappings?: AmpcodeModelMapping[];
|
modelMappings?: AmpcodeModelMapping[];
|
||||||
forceModelMappings?: boolean;
|
forceModelMappings?: boolean;
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -18,6 +18,7 @@ export const LOG_REFRESH_DELAY_MS = 500;
|
|||||||
// 日志相关
|
// 日志相关
|
||||||
export const MAX_LOG_LINES = 2000;
|
export const MAX_LOG_LINES = 2000;
|
||||||
export const LOG_FETCH_LIMIT = 2500;
|
export const LOG_FETCH_LIMIT = 2500;
|
||||||
|
export const LOGS_TIMEOUT_MS = 60 * 1000;
|
||||||
|
|
||||||
// 认证文件分页
|
// 认证文件分页
|
||||||
export const DEFAULT_AUTH_FILES_PAGE_SIZE = 20;
|
export const DEFAULT_AUTH_FILES_PAGE_SIZE = 20;
|
||||||
|
|||||||
Reference in New Issue
Block a user