chore: reformat code

This commit is contained in:
CaIon 2024-03-15 16:05:33 +08:00
parent 25ec99913b
commit d34b55c154
28 changed files with 5856 additions and 5922 deletions

View File

@ -13,7 +13,7 @@ import (
// TODO: when a new api is enabled, check the pricing here // TODO: when a new api is enabled, check the pricing here
// 1 === $0.002 / 1K tokens // 1 === $0.002 / 1K tokens
// 1 === ¥0.014 / 1k tokens // 1 === ¥0.014 / 1k tokens
var ModelRatio = map[string]float64{ var DefaultModelRatio = map[string]float64{
//"midjourney": 50, //"midjourney": 50,
"gpt-4-gizmo-*": 15, "gpt-4-gizmo-*": 15,
"gpt-4": 15, "gpt-4": 15,
@ -115,6 +115,7 @@ var DefaultModelPrice = map[string]float64{
} }
var ModelPrice = map[string]float64{} var ModelPrice = map[string]float64{}
var ModelRatio = map[string]float64{}
func ModelPrice2JSONString() string { func ModelPrice2JSONString() string {
if len(ModelPrice) == 0 { if len(ModelPrice) == 0 {
@ -150,6 +151,9 @@ func GetModelPrice(name string, printErr bool) float64 {
} }
func ModelRatio2JSONString() string { func ModelRatio2JSONString() string {
if len(ModelRatio) == 0 {
ModelRatio = DefaultModelRatio
}
jsonBytes, err := json.Marshal(ModelRatio) jsonBytes, err := json.Marshal(ModelRatio)
if err != nil { if err != nil {
SysError("error marshalling model ratio: " + err.Error()) SysError("error marshalling model ratio: " + err.Error())
@ -163,6 +167,9 @@ func UpdateModelRatioByJSONString(jsonStr string) error {
} }
func GetModelRatio(name string) float64 { func GetModelRatio(name string) float64 {
if len(ModelRatio) == 0 {
ModelRatio = DefaultModelRatio
}
if strings.HasPrefix(name, "gpt-4-gizmo") { if strings.HasPrefix(name, "gpt-4-gizmo") {
name = "gpt-4-gizmo-*" name = "gpt-4-gizmo-*"
} }

View File

@ -49,7 +49,7 @@
] ]
}, },
"devDependencies": { "devDependencies": {
"prettier": "^2.7.1", "prettier": "2.8.8",
"typescript": "4.4.2" "typescript": "4.4.2"
}, },
"prettier": { "prettier": {

View File

@ -8,12 +8,11 @@ import LoginForm from './components/LoginForm';
import NotFound from './pages/NotFound'; import NotFound from './pages/NotFound';
import Setting from './pages/Setting'; import Setting from './pages/Setting';
import EditUser from './pages/User/EditUser'; import EditUser from './pages/User/EditUser';
import { API, getLogo, getSystemName, showError, showNotice } from './helpers'; import { getLogo, getSystemName } from './helpers';
import PasswordResetForm from './components/PasswordResetForm'; import PasswordResetForm from './components/PasswordResetForm';
import GitHubOAuth from './components/GitHubOAuth'; import GitHubOAuth from './components/GitHubOAuth';
import PasswordResetConfirm from './components/PasswordResetConfirm'; import PasswordResetConfirm from './components/PasswordResetConfirm';
import { UserContext } from './context/User'; import { UserContext } from './context/User';
import { StatusContext } from './context/Status';
import Channel from './pages/Channel'; import Channel from './pages/Channel';
import Token from './pages/Token'; import Token from './pages/Token';
import EditChannel from './pages/Channel/EditChannel'; import EditChannel from './pages/Channel/EditChannel';
@ -21,12 +20,13 @@ import Redemption from './pages/Redemption';
import TopUp from './pages/TopUp'; import TopUp from './pages/TopUp';
import Log from './pages/Log'; import Log from './pages/Log';
import Chat from './pages/Chat'; import Chat from './pages/Chat';
import {Layout} from "@douyinfe/semi-ui"; import { Layout } from '@douyinfe/semi-ui';
import Midjourney from "./pages/Midjourney"; import Midjourney from './pages/Midjourney';
import Detail from "./pages/Detail"; import Detail from './pages/Detail';
const Home = lazy(() => import('./pages/Home')); const Home = lazy(() => import('./pages/Home'));
const About = lazy(() => import('./pages/About')); const About = lazy(() => import('./pages/About'));
function App() { function App() {
const [userState, userDispatch] = useContext(UserContext); const [userState, userDispatch] = useContext(UserContext);
// const [statusState, statusDispatch] = useContext(StatusContext); // const [statusState, statusDispatch] = useContext(StatusContext);
@ -47,7 +47,7 @@ function App() {
} }
let logo = getLogo(); let logo = getLogo();
if (logo) { if (logo) {
let linkElement = document.querySelector("link[rel~='icon']"); let linkElement = document.querySelector('link[rel~=\'icon\']');
if (linkElement) { if (linkElement) {
linkElement.href = logo; linkElement.href = logo;
} }
@ -56,185 +56,185 @@ function App() {
return ( return (
<Layout> <Layout>
<Layout.Content> <Layout.Content>
<Routes> <Routes>
<Route <Route
path='/' path="/"
element={ element={
<Suspense fallback={<Loading></Loading>}> <Suspense fallback={<Loading></Loading>}>
<Home /> <Home />
</Suspense> </Suspense>
} }
/> />
<Route <Route
path='/channel' path="/channel"
element={ element={
<PrivateRoute> <PrivateRoute>
<Channel /> <Channel />
</PrivateRoute> </PrivateRoute>
} }
/> />
<Route <Route
path='/channel/edit/:id' path="/channel/edit/:id"
element={ element={
<Suspense fallback={<Loading></Loading>}> <Suspense fallback={<Loading></Loading>}>
<EditChannel /> <EditChannel />
</Suspense> </Suspense>
} }
/> />
<Route <Route
path='/channel/add' path="/channel/add"
element={ element={
<Suspense fallback={<Loading></Loading>}> <Suspense fallback={<Loading></Loading>}>
<EditChannel /> <EditChannel />
</Suspense> </Suspense>
} }
/> />
<Route <Route
path='/token' path="/token"
element={ element={
<PrivateRoute> <PrivateRoute>
<Token /> <Token />
</PrivateRoute> </PrivateRoute>
} }
/> />
<Route <Route
path='/redemption' path="/redemption"
element={ element={
<PrivateRoute> <PrivateRoute>
<Redemption /> <Redemption />
</PrivateRoute> </PrivateRoute>
} }
/> />
<Route <Route
path='/user' path="/user"
element={ element={
<PrivateRoute> <PrivateRoute>
<User /> <User />
</PrivateRoute> </PrivateRoute>
} }
/> />
<Route <Route
path='/user/edit/:id' path="/user/edit/:id"
element={ element={
<Suspense fallback={<Loading></Loading>}> <Suspense fallback={<Loading></Loading>}>
<EditUser /> <EditUser />
</Suspense> </Suspense>
} }
/> />
<Route <Route
path='/user/edit' path="/user/edit"
element={ element={
<Suspense fallback={<Loading></Loading>}> <Suspense fallback={<Loading></Loading>}>
<EditUser /> <EditUser />
</Suspense> </Suspense>
} }
/> />
<Route <Route
path='/user/reset' path="/user/reset"
element={ element={
<Suspense fallback={<Loading></Loading>}> <Suspense fallback={<Loading></Loading>}>
<PasswordResetConfirm /> <PasswordResetConfirm />
</Suspense> </Suspense>
} }
/> />
<Route <Route
path='/login' path="/login"
element={ element={
<Suspense fallback={<Loading></Loading>}> <Suspense fallback={<Loading></Loading>}>
<LoginForm /> <LoginForm />
</Suspense> </Suspense>
} }
/> />
<Route <Route
path='/register' path="/register"
element={ element={
<Suspense fallback={<Loading></Loading>}> <Suspense fallback={<Loading></Loading>}>
<RegisterForm /> <RegisterForm />
</Suspense> </Suspense>
} }
/> />
<Route <Route
path='/reset' path="/reset"
element={ element={
<Suspense fallback={<Loading></Loading>}> <Suspense fallback={<Loading></Loading>}>
<PasswordResetForm /> <PasswordResetForm />
</Suspense> </Suspense>
} }
/> />
<Route <Route
path='/oauth/github' path="/oauth/github"
element={ element={
<Suspense fallback={<Loading></Loading>}> <Suspense fallback={<Loading></Loading>}>
<GitHubOAuth /> <GitHubOAuth />
</Suspense> </Suspense>
} }
/> />
<Route <Route
path='/setting' path="/setting"
element={ element={
<PrivateRoute> <PrivateRoute>
<Suspense fallback={<Loading></Loading>}> <Suspense fallback={<Loading></Loading>}>
<Setting /> <Setting />
</Suspense> </Suspense>
</PrivateRoute> </PrivateRoute>
} }
/> />
<Route <Route
path='/topup' path="/topup"
element={ element={
<PrivateRoute> <PrivateRoute>
<Suspense fallback={<Loading></Loading>}> <Suspense fallback={<Loading></Loading>}>
<TopUp /> <TopUp />
</Suspense> </Suspense>
</PrivateRoute> </PrivateRoute>
} }
/> />
<Route <Route
path='/log' path="/log"
element={ element={
<PrivateRoute> <PrivateRoute>
<Log /> <Log />
</PrivateRoute> </PrivateRoute>
} }
/> />
<Route <Route
path='/detail' path="/detail"
element={ element={
<PrivateRoute> <PrivateRoute>
<Detail /> <Detail />
</PrivateRoute> </PrivateRoute>
} }
/> />
<Route <Route
path='/midjourney' path="/midjourney"
element={ element={
<PrivateRoute> <PrivateRoute>
<Midjourney /> <Midjourney />
</PrivateRoute> </PrivateRoute>
} }
/> />
<Route <Route
path='/about' path="/about"
element={ element={
<Suspense fallback={<Loading></Loading>}> <Suspense fallback={<Loading></Loading>}>
<About /> <About />
</Suspense> </Suspense>
} }
/> />
<Route <Route
path='/chat' path="/chat"
element={ element={
<Suspense fallback={<Loading></Loading>}> <Suspense fallback={<Loading></Loading>}>
<Chat /> <Chat />
</Suspense> </Suspense>
} }
/> />
<Route path='*' element={ <Route path="*" element={
<NotFound /> <NotFound />
} /> } />
</Routes> </Routes>
</Layout.Content> </Layout.Content>
</Layout> </Layout>
); );
} }

File diff suppressed because it is too large Load Diff

View File

@ -1,7 +1,7 @@
import React, { useEffect, useState } from 'react'; import React, { useEffect, useState } from 'react';
import { getFooterHTML, getSystemName } from '../helpers'; import { getFooterHTML, getSystemName } from '../helpers';
import {Layout} from "@douyinfe/semi-ui"; import { Layout } from '@douyinfe/semi-ui';
const Footer = () => { const Footer = () => {
const systemName = getSystemName(); const systemName = getSystemName();
@ -29,30 +29,30 @@ const Footer = () => {
return ( return (
<Layout> <Layout>
<Layout.Content style={{textAlign: 'center'}}> <Layout.Content style={{ textAlign: 'center' }}>
{footer ? ( {footer ? (
<div <div
className='custom-footer' className="custom-footer"
dangerouslySetInnerHTML={{ __html: footer }} dangerouslySetInnerHTML={{ __html: footer }}
></div> ></div>
) : ( ) : (
<div className='custom-footer'> <div className="custom-footer">
<a <a
href='https://github.com/Calcium-Ion/new-api' href="https://github.com/Calcium-Ion/new-api"
target='_blank' target="_blank" rel="noreferrer"
> >
New API {process.env.REACT_APP_VERSION}{' '} New API {process.env.REACT_APP_VERSION}{' '}
</a> </a>
{' '} {' '}
<a href='https://github.com/Calcium-Ion' target='_blank'> <a href="https://github.com/Calcium-Ion" target="_blank" rel="noreferrer">
Calcium-Ion Calcium-Ion
</a>{' '} </a>{' '}
开发基于{' '} 开发基于{' '}
<a href='https://github.com/songquanpeng/one-api' target='_blank'> <a href="https://github.com/songquanpeng/one-api" target="_blank" rel="noreferrer">
One API v0.5.4 One API v0.5.4
</a>{' '} </a>{' '}
本项目根据{' '} 本项目根据{' '}
<a href='https://opensource.org/licenses/mit-license.php'> <a href="https://opensource.org/licenses/mit-license.php">
MIT 许可证 MIT 许可证
</a>{' '} </a>{' '}
授权 授权

View File

@ -49,7 +49,7 @@ const GitHubOAuth = () => {
return ( return (
<Segment style={{ minHeight: '300px' }}> <Segment style={{ minHeight: '300px' }}>
<Dimmer active inverted> <Dimmer active inverted>
<Loader size='large'>{prompt}</Loader> <Loader size="large">{prompt}</Loader>
</Dimmer> </Dimmer>
</Segment> </Segment>
); );

View File

@ -1,165 +1,161 @@
import React, {useContext, useEffect, useRef, useState} from 'react'; import React, { useContext, useEffect, useState } from 'react';
import {Link, useNavigate} from 'react-router-dom'; import { Link, useNavigate } from 'react-router-dom';
import {UserContext} from '../context/User'; import { UserContext } from '../context/User';
import {API, getLogo, getSystemName, isAdmin, isMobile, showSuccess} from '../helpers'; import { API, getLogo, getSystemName, showSuccess } from '../helpers';
import '../index.css'; import '../index.css';
import fireworks from 'react-fireworks'; import fireworks from 'react-fireworks';
import { import { IconHelpCircle, IconKey, IconUser } from '@douyinfe/semi-icons';
IconKey, import { Avatar, Dropdown, Layout, Nav, Switch } from '@douyinfe/semi-ui';
IconUser, import { stringToColor } from '../helpers/render';
IconHelpCircle
} from '@douyinfe/semi-icons';
import {Nav, Avatar, Dropdown, Layout, Switch} from '@douyinfe/semi-ui';
import {stringToColor} from "../helpers/render";
// HeaderBar Buttons // HeaderBar Buttons
let headerButtons = [ let headerButtons = [
{ {
text: '关于', text: '关于',
itemKey: 'about', itemKey: 'about',
to: '/about', to: '/about',
icon: <IconHelpCircle/> icon: <IconHelpCircle />
}, }
]; ];
if (localStorage.getItem('chat_link')) { if (localStorage.getItem('chat_link')) {
headerButtons.splice(1, 0, { headerButtons.splice(1, 0, {
name: '聊天', name: '聊天',
to: '/chat', to: '/chat',
icon: 'comments' icon: 'comments'
}); });
} }
const HeaderBar = () => { const HeaderBar = () => {
const [userState, userDispatch] = useContext(UserContext); const [userState, userDispatch] = useContext(UserContext);
let navigate = useNavigate(); let navigate = useNavigate();
const [showSidebar, setShowSidebar] = useState(false); const [showSidebar, setShowSidebar] = useState(false);
const [dark, setDark] = useState(false); const [dark, setDark] = useState(false);
const systemName = getSystemName(); const systemName = getSystemName();
const logo = getLogo(); const logo = getLogo();
var themeMode = localStorage.getItem('theme-mode'); var themeMode = localStorage.getItem('theme-mode');
const currentDate = new Date(); const currentDate = new Date();
// enable fireworks on new year(1.1 and 2.9-2.24) // enable fireworks on new year(1.1 and 2.9-2.24)
const isNewYear = (currentDate.getMonth() === 0 && currentDate.getDate() === 1) || (currentDate.getMonth() === 1 && currentDate.getDate() >= 9 && currentDate.getDate() <= 24); const isNewYear = (currentDate.getMonth() === 0 && currentDate.getDate() === 1) || (currentDate.getMonth() === 1 && currentDate.getDate() >= 9 && currentDate.getDate() <= 24);
async function logout() { async function logout() {
setShowSidebar(false); setShowSidebar(false);
await API.get('/api/user/logout'); await API.get('/api/user/logout');
showSuccess('注销成功!'); showSuccess('注销成功!');
userDispatch({type: 'logout'}); userDispatch({ type: 'logout' });
localStorage.removeItem('user'); localStorage.removeItem('user');
navigate('/login'); navigate('/login');
}
const handleNewYearClick = () => {
fireworks.init('root', {});
fireworks.start();
setTimeout(() => {
fireworks.stop();
setTimeout(() => {
window.location.reload();
}, 10000);
}, 3000);
};
useEffect(() => {
if (themeMode === 'dark') {
switchMode(true);
} }
if (isNewYear) {
console.log('Happy New Year!');
}
}, []);
const handleNewYearClick = () => { const switchMode = (model) => {
fireworks.init("root",{}); const body = document.body;
fireworks.start(); if (!model) {
setTimeout(() => { body.removeAttribute('theme-mode');
fireworks.stop(); localStorage.setItem('theme-mode', 'light');
setTimeout(() => { } else {
window.location.reload(); body.setAttribute('theme-mode', 'dark');
}, 10000); localStorage.setItem('theme-mode', 'dark');
}, 3000); }
}; setDark(model);
};
return (
<>
<Layout>
<div style={{ width: '100%' }}>
<Nav
mode={'horizontal'}
// bodyStyle={{ height: 100 }}
renderWrapper={({ itemElement, isSubNav, isInSubNav, props }) => {
const routerMap = {
about: '/about',
login: '/login',
register: '/register'
};
return (
<Link
style={{ textDecoration: 'none' }}
to={routerMap[props.itemKey]}
>
{itemElement}
</Link>
);
}}
selectedKeys={[]}
// items={headerButtons}
onSelect={key => {
useEffect(() => { }}
if (themeMode === 'dark') { footer={
switchMode(true); <>
} {isNewYear &&
if (isNewYear) { // happy new year
console.log('Happy New Year!'); <Dropdown
} position="bottomRight"
}, []); render={
<Dropdown.Menu>
const switchMode = (model) => { <Dropdown.Item onClick={handleNewYearClick}>Happy New Year!!!</Dropdown.Item>
const body = document.body; </Dropdown.Menu>
if (!model) { }
body.removeAttribute('theme-mode'); >
localStorage.setItem('theme-mode', 'light'); <Nav.Item itemKey={'new-year'} text={'🏮'} />
} else { </Dropdown>
body.setAttribute('theme-mode', 'dark'); }
localStorage.setItem('theme-mode', 'dark'); <Nav.Item itemKey={'about'} icon={<IconHelpCircle />} />
} <Switch checkedText="🌞" size={'large'} checked={dark} uncheckedText="🌙" onChange={switchMode} />
setDark(model); {userState.user ?
}; <>
return ( <Dropdown
<> position="bottomRight"
<Layout> render={
<div style={{width: '100%'}}> <Dropdown.Menu>
<Nav <Dropdown.Item onClick={logout}>退出</Dropdown.Item>
mode={'horizontal'} </Dropdown.Menu>
// bodyStyle={{ height: 100 }} }
renderWrapper={({itemElement, isSubNav, isInSubNav, props}) => {
const routerMap = {
about: "/about",
login: "/login",
register: "/register",
};
return (
<Link
style={{textDecoration: "none"}}
to={routerMap[props.itemKey]}
>
{itemElement}
</Link>
);
}}
selectedKeys={[]}
// items={headerButtons}
onSelect={key => {
}}
footer={
<>
{isNewYear &&
// happy new year
<Dropdown
position="bottomRight"
render={
<Dropdown.Menu>
<Dropdown.Item onClick={handleNewYearClick}>Happy New Year!!!</Dropdown.Item>
</Dropdown.Menu>
}
>
<Nav.Item itemKey={'new-year'} text={'🏮'}/>
</Dropdown>
}
<Nav.Item itemKey={'about'} icon={<IconHelpCircle />} />
<Switch checkedText="🌞" size={'large'} checked={dark} uncheckedText="🌙" onChange={switchMode} />
{userState.user ?
<>
<Dropdown
position="bottomRight"
render={
<Dropdown.Menu>
<Dropdown.Item onClick={logout}>退出</Dropdown.Item>
</Dropdown.Menu>
}
>
<Avatar size="small" color={stringToColor(userState.user.username)} style={{ margin: 4 }}>
{userState.user.username[0]}
</Avatar>
<span>{userState.user.username}</span>
</Dropdown>
</>
:
<>
<Nav.Item itemKey={'login'} text={'登录'} icon={<IconKey />} />
<Nav.Item itemKey={'register'} text={'注册'} icon={<IconUser />} />
</>
}
</>
}
> >
</Nav> <Avatar size="small" color={stringToColor(userState.user.username)} style={{ margin: 4 }}>
</div> {userState.user.username[0]}
</Layout> </Avatar>
</> <span>{userState.user.username}</span>
); </Dropdown>
</>
:
<>
<Nav.Item itemKey={'login'} text={'登录'} icon={<IconKey />} />
<Nav.Item itemKey={'register'} text={'注册'} icon={<IconUser />} />
</>
}
</>
}
>
</Nav>
</div>
</Layout>
</>
);
}; };
export default HeaderBar; export default HeaderBar;

View File

@ -1,5 +1,5 @@
import React from 'react'; import React from 'react';
import { Segment, Dimmer, Loader } from 'semantic-ui-react'; import { Dimmer, Loader, Segment } from 'semantic-ui-react';
const Loading = ({ prompt: name = 'page' }) => { const Loading = ({ prompt: name = 'page' }) => {
return ( return (

View File

@ -1,254 +1,254 @@
import React, { useContext, useEffect, useState } from 'react'; import React, { useContext, useEffect, useState } from 'react';
import { Link, useNavigate, useSearchParams } from 'react-router-dom'; import { Link, useNavigate, useSearchParams } from 'react-router-dom';
import { UserContext } from '../context/User'; import { UserContext } from '../context/User';
import { API, getLogo, isMobile, showError, showInfo, showSuccess, showWarning } from '../helpers'; import { API, getLogo, showError, showInfo, showSuccess } from '../helpers';
import { onGitHubOAuthClicked } from './utils'; import { onGitHubOAuthClicked } from './utils';
import Turnstile from "react-turnstile"; import Turnstile from 'react-turnstile';
import { Layout, Card, Image, Form, Button, Divider, Modal, Icon } from '@douyinfe/semi-ui'; import { Button, Card, Divider, Form, Icon, Layout, Modal } from '@douyinfe/semi-ui';
import Title from "@douyinfe/semi-ui/lib/es/typography/title"; import Title from '@douyinfe/semi-ui/lib/es/typography/title';
import Text from "@douyinfe/semi-ui/lib/es/typography/text"; import Text from '@douyinfe/semi-ui/lib/es/typography/text';
import TelegramLoginButton from 'react-telegram-login'; import TelegramLoginButton from 'react-telegram-login';
import { IconGithubLogo } from '@douyinfe/semi-icons'; import { IconGithubLogo } from '@douyinfe/semi-icons';
import WeChatIcon from './WeChatIcon'; import WeChatIcon from './WeChatIcon';
const LoginForm = () => { const LoginForm = () => {
const [inputs, setInputs] = useState({ const [inputs, setInputs] = useState({
username: '', username: '',
password: '', password: '',
wechat_verification_code: '' wechat_verification_code: ''
}); });
const [searchParams, setSearchParams] = useSearchParams(); const [searchParams, setSearchParams] = useSearchParams();
const [submitted, setSubmitted] = useState(false); const [submitted, setSubmitted] = useState(false);
const { username, password } = inputs; const { username, password } = inputs;
const [userState, userDispatch] = useContext(UserContext); const [userState, userDispatch] = useContext(UserContext);
const [turnstileEnabled, setTurnstileEnabled] = useState(false); const [turnstileEnabled, setTurnstileEnabled] = useState(false);
const [turnstileSiteKey, setTurnstileSiteKey] = useState(''); const [turnstileSiteKey, setTurnstileSiteKey] = useState('');
const [turnstileToken, setTurnstileToken] = useState(''); const [turnstileToken, setTurnstileToken] = useState('');
let navigate = useNavigate(); let navigate = useNavigate();
const [status, setStatus] = useState({}); const [status, setStatus] = useState({});
const logo = getLogo(); const logo = getLogo();
useEffect(() => { useEffect(() => {
if (searchParams.get('expired')) { if (searchParams.get('expired')) {
showError('未登录或登录已过期,请重新登录!'); showError('未登录或登录已过期,请重新登录!');
}
let status = localStorage.getItem('status');
if (status) {
status = JSON.parse(status);
setStatus(status);
if (status.turnstile_check) {
setTurnstileEnabled(true);
setTurnstileSiteKey(status.turnstile_site_key);
}
}
}, []);
const [showWeChatLoginModal, setShowWeChatLoginModal] = useState(false);
const onWeChatLoginClicked = () => {
setShowWeChatLoginModal(true);
};
const onSubmitWeChatVerificationCode = async () => {
if (turnstileEnabled && turnstileToken === '') {
showInfo('请稍后几秒重试Turnstile 正在检查用户环境!');
return;
}
const res = await API.get(
`/api/oauth/wechat?code=${inputs.wechat_verification_code}`
);
const { success, message, data } = res.data;
if (success) {
userDispatch({ type: 'login', payload: data });
localStorage.setItem('user', JSON.stringify(data));
navigate('/');
showSuccess('登录成功!');
setShowWeChatLoginModal(false);
} else {
showError(message);
}
};
function handleChange(name, value) {
setInputs((inputs) => ({ ...inputs, [name]: value }));
} }
let status = localStorage.getItem('status');
async function handleSubmit(e) { if (status) {
if (turnstileEnabled && turnstileToken === '') { status = JSON.parse(status);
showInfo('请稍后几秒重试Turnstile 正在检查用户环境!'); setStatus(status);
return; if (status.turnstile_check) {
} setTurnstileEnabled(true);
setSubmitted(true); setTurnstileSiteKey(status.turnstile_site_key);
if (username && password) { }
const res = await API.post(`/api/user/login?turnstile=${turnstileToken}`, {
username,
password
});
const { success, message, data } = res.data;
if (success) {
userDispatch({ type: 'login', payload: data });
localStorage.setItem('user', JSON.stringify(data));
showSuccess('登录成功!');
if (username === 'root' && password === '123456') {
Modal.error({ title: '您正在使用默认密码!', content: '请立刻修改默认密码!', centered: true });
}
navigate('/token');
} else {
showError(message);
}
} else {
showError('请输入用户名和密码!');
}
} }
}, []);
// 添加Telegram登录处理函数 const [showWeChatLoginModal, setShowWeChatLoginModal] = useState(false);
const onTelegramLoginClicked = async (response) => {
const fields = ["id", "first_name", "last_name", "username", "photo_url", "auth_date", "hash", "lang"];
const params = {};
fields.forEach((field) => {
if (response[field]) {
params[field] = response[field];
}
});
const res = await API.get(`/api/oauth/telegram/login`, { params });
const { success, message, data } = res.data;
if (success) {
userDispatch({ type: 'login', payload: data });
localStorage.setItem('user', JSON.stringify(data));
showSuccess('登录成功!');
navigate('/');
} else {
showError(message);
}
};
return ( const onWeChatLoginClicked = () => {
<div> setShowWeChatLoginModal(true);
<Layout> };
<Layout.Header>
</Layout.Header>
<Layout.Content>
<div style={{ justifyContent: 'center', display: "flex", marginTop: 120 }}>
<div style={{ width: 500 }}>
<Card>
<Title heading={2} style={{ textAlign: 'center' }}>
用户登录
</Title>
<Form>
<Form.Input
field={'username'}
label={'用户名'}
placeholder='用户名'
name='username'
onChange={(value) => handleChange('username', value)}
/>
<Form.Input
field={'password'}
label={'密码'}
placeholder='密码'
name='password'
type='password'
onChange={(value) => handleChange('password', value)}
/>
<Button theme='solid' style={{ width: '100%' }} type={'primary'} size='large' const onSubmitWeChatVerificationCode = async () => {
htmlType={'submit'} onClick={handleSubmit}> if (turnstileEnabled && turnstileToken === '') {
登录 showInfo('请稍后几秒重试Turnstile 正在检查用户环境!');
</Button> return;
</Form> }
<div style={{ display: 'flex', justifyContent: 'space-between', marginTop: 20 }}> const res = await API.get(
<Text> `/api/oauth/wechat?code=${inputs.wechat_verification_code}`
没有账号请先 <Link to='/register'>注册账号</Link>
</Text>
<Text>
忘记密码 <Link to='/reset'>点击重置</Link>
</Text>
</div>
{status.github_oauth || status.wechat_login || status.telegram_oauth ? (
<>
<Divider margin='12px' align='center'>
第三方登录
</Divider>
<div style={{ display: 'flex', justifyContent: 'center', marginTop: 20 }}>
{status.github_oauth ? (
<Button
type='primary'
icon={<IconGithubLogo />}
onClick={() => onGitHubOAuthClicked(status.github_client_id)}
/>
) : (
<></>
)}
{status.wechat_login ? (
<Button
type='primary'
style={{color: 'rgba(var(--semi-green-5), 1)'}}
icon={<Icon svg={<WeChatIcon />} />}
onClick={onWeChatLoginClicked}
/>
) : (
<></>
)}
{status.telegram_oauth ? (
<TelegramLoginButton dataOnauth={onTelegramLoginClicked} botName={status.telegram_bot_name} />
) : (
<></>
)}
</div>
</>
) : (
<></>
)}
<Modal
title="微信扫码登录"
visible={showWeChatLoginModal}
maskClosable={true}
onOk={onSubmitWeChatVerificationCode}
onCancel={() => setShowWeChatLoginModal(false)}
okText={'登录'}
size={'small'}
centered={true}
>
<div style={{ display: 'flex', alignItem: 'center', flexDirection: 'column' }}>
<img src={status.wechat_qrcode}/>
</div>
<div style={{textAlign: 'center'}}>
<p>
微信扫码关注公众号输入验证码获取验证码三分钟内有效
</p>
</div>
<Form size='large'>
<Form.Input
field={'wechat_verification_code'}
placeholder='验证码'
label={'验证码'}
value={inputs.wechat_verification_code}
onChange={(value) => handleChange('wechat_verification_code', value)}
/>
</Form>
</Modal>
</Card>
{turnstileEnabled ? (
<div style={{ display: 'flex', justifyContent: 'center', marginTop: 20 }}>
<Turnstile
sitekey={turnstileSiteKey}
onVerify={(token) => {
setTurnstileToken(token);
}}
/>
</div>
) : (
<></>
)}
</div>
</div>
</Layout.Content>
</Layout>
</div>
); );
const { success, message, data } = res.data;
if (success) {
userDispatch({ type: 'login', payload: data });
localStorage.setItem('user', JSON.stringify(data));
navigate('/');
showSuccess('登录成功!');
setShowWeChatLoginModal(false);
} else {
showError(message);
}
};
function handleChange(name, value) {
setInputs((inputs) => ({ ...inputs, [name]: value }));
}
async function handleSubmit(e) {
if (turnstileEnabled && turnstileToken === '') {
showInfo('请稍后几秒重试Turnstile 正在检查用户环境!');
return;
}
setSubmitted(true);
if (username && password) {
const res = await API.post(`/api/user/login?turnstile=${turnstileToken}`, {
username,
password
});
const { success, message, data } = res.data;
if (success) {
userDispatch({ type: 'login', payload: data });
localStorage.setItem('user', JSON.stringify(data));
showSuccess('登录成功!');
if (username === 'root' && password === '123456') {
Modal.error({ title: '您正在使用默认密码!', content: '请立刻修改默认密码!', centered: true });
}
navigate('/token');
} else {
showError(message);
}
} else {
showError('请输入用户名和密码!');
}
}
// 添加Telegram登录处理函数
const onTelegramLoginClicked = async (response) => {
const fields = ['id', 'first_name', 'last_name', 'username', 'photo_url', 'auth_date', 'hash', 'lang'];
const params = {};
fields.forEach((field) => {
if (response[field]) {
params[field] = response[field];
}
});
const res = await API.get(`/api/oauth/telegram/login`, { params });
const { success, message, data } = res.data;
if (success) {
userDispatch({ type: 'login', payload: data });
localStorage.setItem('user', JSON.stringify(data));
showSuccess('登录成功!');
navigate('/');
} else {
showError(message);
}
};
return (
<div>
<Layout>
<Layout.Header>
</Layout.Header>
<Layout.Content>
<div style={{ justifyContent: 'center', display: 'flex', marginTop: 120 }}>
<div style={{ width: 500 }}>
<Card>
<Title heading={2} style={{ textAlign: 'center' }}>
用户登录
</Title>
<Form>
<Form.Input
field={'username'}
label={'用户名'}
placeholder="用户名"
name="username"
onChange={(value) => handleChange('username', value)}
/>
<Form.Input
field={'password'}
label={'密码'}
placeholder="密码"
name="password"
type="password"
onChange={(value) => handleChange('password', value)}
/>
<Button theme="solid" style={{ width: '100%' }} type={'primary'} size="large"
htmlType={'submit'} onClick={handleSubmit}>
登录
</Button>
</Form>
<div style={{ display: 'flex', justifyContent: 'space-between', marginTop: 20 }}>
<Text>
没有账号请先 <Link to="/register">注册账号</Link>
</Text>
<Text>
忘记密码 <Link to="/reset">点击重置</Link>
</Text>
</div>
{status.github_oauth || status.wechat_login || status.telegram_oauth ? (
<>
<Divider margin="12px" align="center">
第三方登录
</Divider>
<div style={{ display: 'flex', justifyContent: 'center', marginTop: 20 }}>
{status.github_oauth ? (
<Button
type="primary"
icon={<IconGithubLogo />}
onClick={() => onGitHubOAuthClicked(status.github_client_id)}
/>
) : (
<></>
)}
{status.wechat_login ? (
<Button
type="primary"
style={{ color: 'rgba(var(--semi-green-5), 1)' }}
icon={<Icon svg={<WeChatIcon />} />}
onClick={onWeChatLoginClicked}
/>
) : (
<></>
)}
{status.telegram_oauth ? (
<TelegramLoginButton dataOnauth={onTelegramLoginClicked} botName={status.telegram_bot_name} />
) : (
<></>
)}
</div>
</>
) : (
<></>
)}
<Modal
title="微信扫码登录"
visible={showWeChatLoginModal}
maskClosable={true}
onOk={onSubmitWeChatVerificationCode}
onCancel={() => setShowWeChatLoginModal(false)}
okText={'登录'}
size={'small'}
centered={true}
>
<div style={{ display: 'flex', alignItem: 'center', flexDirection: 'column' }}>
<img src={status.wechat_qrcode} />
</div>
<div style={{ textAlign: 'center' }}>
<p>
微信扫码关注公众号输入验证码获取验证码三分钟内有效
</p>
</div>
<Form size="large">
<Form.Input
field={'wechat_verification_code'}
placeholder="验证码"
label={'验证码'}
value={inputs.wechat_verification_code}
onChange={(value) => handleChange('wechat_verification_code', value)}
/>
</Form>
</Modal>
</Card>
{turnstileEnabled ? (
<div style={{ display: 'flex', justifyContent: 'center', marginTop: 20 }}>
<Turnstile
sitekey={turnstileSiteKey}
onVerify={(token) => {
setTurnstileToken(token);
}}
/>
</div>
) : (
<></>
)}
</div>
</div>
</Layout.Content>
</Layout>
</div>
);
}; };
export default LoginForm; export default LoginForm;

View File

@ -1,493 +1,399 @@
import React, {useEffect, useState} from 'react'; import React, { useEffect, useState } from 'react';
import {API, copy, isAdmin, showError, showSuccess, timestamp2string} from '../helpers'; import { API, copy, isAdmin, showError, showSuccess, timestamp2string } from '../helpers';
import {Table, Avatar, Tag, Form, Button, Layout, Select, Popover, Modal, Spin, Space} from '@douyinfe/semi-ui'; import { Avatar, Button, Form, Layout, Modal, Select, Space, Spin, Table, Tag } from '@douyinfe/semi-ui';
import {ITEMS_PER_PAGE} from '../constants'; import { ITEMS_PER_PAGE } from '../constants';
import {renderNumber, renderQuota, stringToColor} from '../helpers/render'; import { renderNumber, renderQuota, stringToColor } from '../helpers/render';
import Paragraph from "@douyinfe/semi-ui/lib/es/typography/paragraph"; import Paragraph from '@douyinfe/semi-ui/lib/es/typography/paragraph';
const {Header} = Layout; const { Header } = Layout;
function renderTimestamp(timestamp) { function renderTimestamp(timestamp) {
return ( return (<>
<> {timestamp2string(timestamp)}
{timestamp2string(timestamp)} </>);
</>
);
} }
const MODE_OPTIONS = [ const MODE_OPTIONS = [{ key: 'all', text: '全部用户', value: 'all' }, { key: 'self', text: '当前用户', value: 'self' }];
{key: 'all', text: '全部用户', value: 'all'},
{key: 'self', text: '当前用户', value: 'self'}
];
const colors = ['amber', 'blue', 'cyan', 'green', 'grey', 'indigo', const colors = ['amber', 'blue', 'cyan', 'green', 'grey', 'indigo', 'light-blue', 'lime', 'orange', 'pink', 'purple', 'red', 'teal', 'violet', 'yellow'];
'light-blue', 'lime', 'orange', 'pink',
'purple', 'red', 'teal', 'violet', 'yellow'
]
function renderType(type) { function renderType(type) {
switch (type) { switch (type) {
case 1: case 1:
return <Tag color='cyan' size='large'> 充值 </Tag>; return <Tag color="cyan" size="large"> 充值 </Tag>;
case 2: case 2:
return <Tag color='lime' size='large'> 消费 </Tag>; return <Tag color="lime" size="large"> 消费 </Tag>;
case 3: case 3:
return <Tag color='orange' size='large'> 管理 </Tag>; return <Tag color="orange" size="large"> 管理 </Tag>;
case 4: case 4:
return <Tag color='purple' size='large'> 系统 </Tag>; return <Tag color="purple" size="large"> 系统 </Tag>;
default: default:
return <Tag color='black' size='large'> 未知 </Tag>; return <Tag color="black" size="large"> 未知 </Tag>;
} }
} }
function renderIsStream(bool) { function renderIsStream(bool) {
if (bool) { if (bool) {
return <Tag color='blue' size='large'></Tag>; return <Tag color="blue" size="large"></Tag>;
} else { } else {
return <Tag color='purple' size='large'>非流</Tag>; return <Tag color="purple" size="large">非流</Tag>;
} }
} }
function renderUseTime(type) { function renderUseTime(type) {
const time = parseInt(type); const time = parseInt(type);
if (time < 101) { if (time < 101) {
return <Tag color='green' size='large'> {time} s </Tag>; return <Tag color="green" size="large"> {time} s </Tag>;
} else if (time < 300) { } else if (time < 300) {
return <Tag color='orange' size='large'> {time} s </Tag>; return <Tag color="orange" size="large"> {time} s </Tag>;
} else { } else {
return <Tag color='red' size='large'> {time} s </Tag>; return <Tag color="red" size="large"> {time} s </Tag>;
} }
} }
const LogsTable = () => { const LogsTable = () => {
const columns = [ const columns = [{
{ title: '时间', dataIndex: 'timestamp2string'
title: '时间', }, {
dataIndex: 'timestamp2string', title: '渠道',
}, dataIndex: 'channel',
{ className: isAdmin() ? 'tableShow' : 'tableHiddle',
title: '渠道', render: (text, record, index) => {
dataIndex: 'channel', return (isAdminUser ? record.type === 0 || record.type === 2 ? <div>
className: isAdmin() ? 'tableShow' : 'tableHiddle', {<Tag color={colors[parseInt(text) % colors.length]} size="large"> {text} </Tag>}
render: (text, record, index) => { </div> : <></> : <></>);
return (
isAdminUser ?
record.type === 0 || record.type === 2 ?
<div>
{<Tag color={colors[parseInt(text) % colors.length]} size='large'> {text} </Tag>}
</div>
:
<></>
:
<></>
);
},
},
{
title: '用户',
dataIndex: 'username',
className: isAdmin() ? 'tableShow' : 'tableHiddle',
render: (text, record, index) => {
return (
isAdminUser ?
<div>
<Avatar size="small" color={stringToColor(text)} style={{marginRight: 4}}
onClick={() => showUserInfo(record.user_id)}>
{typeof text === 'string' && text.slice(0, 1)}
</Avatar>
{text}
</div>
:
<></>
);
},
},
{
title: '令牌',
dataIndex: 'token_name',
render: (text, record, index) => {
return (
record.type === 0 || record.type === 2 ?
<div>
<Tag color='grey' size='large' onClick={() => {
copyText(text)
}}> {text} </Tag>
</div>
:
<></>
);
},
},
{
title: '类型',
dataIndex: 'type',
render: (text, record, index) => {
return (
<div>
{renderType(text)}
</div>
);
},
},
{
title: '模型',
dataIndex: 'model_name',
render: (text, record, index) => {
return (
record.type === 0 || record.type === 2 ?
<div>
<Tag color={stringToColor(text)} size='large' onClick={() => {
copyText(text)
}}> {text} </Tag>
</div>
:
<></>
);
},
},
{
title: '用时',
dataIndex: 'use_time',
render: (text, record, index) => {
return (
<div>
<Space>
{renderUseTime(text)}
{renderIsStream(record.is_stream)}
</Space>
</div>
);
},
},
{
title: '提示',
dataIndex: 'prompt_tokens',
render: (text, record, index) => {
return (
record.type === 0 || record.type === 2 ?
<div>
{<span> {text} </span>}
</div>
:
<></>
);
},
},
{
title: '补全',
dataIndex: 'completion_tokens',
render: (text, record, index) => {
return (
parseInt(text) > 0 && (record.type === 0 || record.type === 2) ?
<div>
{<span> {text} </span>}
</div>
:
<></>
);
},
},
{
title: '花费',
dataIndex: 'quota',
render: (text, record, index) => {
return (
record.type === 0 || record.type === 2 ?
<div>
{
renderQuota(text, 6)
}
</div>
:
<></>
);
}
},
{
title: '详情',
dataIndex: 'content',
render: (text, record, index) => {
return <Paragraph ellipsis={{rows: 2, showTooltip: {type: 'popover', opts: {style: {width: 240}}}}}
style={{maxWidth: 240}}>
{text}
</Paragraph>
}
}
];
const [logs, setLogs] = useState([]);
const [showStat, setShowStat] = useState(false);
const [loading, setLoading] = useState(false);
const [loadingStat, setLoadingStat] = useState(false);
const [activePage, setActivePage] = useState(1);
const [logCount, setLogCount] = useState(ITEMS_PER_PAGE);
const [pageSize, setPageSize] = useState(ITEMS_PER_PAGE);
const [searchKeyword, setSearchKeyword] = useState('');
const [searching, setSearching] = useState(false);
const [logType, setLogType] = useState(0);
const isAdminUser = isAdmin();
let now = new Date();
// 初始化start_timestamp为前一天
const [inputs, setInputs] = useState({
username: '',
token_name: '',
model_name: '',
start_timestamp: timestamp2string(now.getTime() / 1000 - 86400),
end_timestamp: timestamp2string(now.getTime() / 1000 + 3600),
channel: ''
});
const {username, token_name, model_name, start_timestamp, end_timestamp, channel} = inputs;
const [stat, setStat] = useState({
quota: 0,
token: 0
});
const handleInputChange = (value, name) => {
setInputs((inputs) => ({...inputs, [name]: value}));
};
const getLogSelfStat = async () => {
let localStartTimestamp = Date.parse(start_timestamp) / 1000;
let localEndTimestamp = Date.parse(end_timestamp) / 1000;
let res = await API.get(`/api/log/self/stat?type=${logType}&token_name=${token_name}&model_name=${model_name}&start_timestamp=${localStartTimestamp}&end_timestamp=${localEndTimestamp}`);
const {success, message, data} = res.data;
if (success) {
setStat(data);
} else {
showError(message);
}
};
const getLogStat = async () => {
let localStartTimestamp = Date.parse(start_timestamp) / 1000;
let localEndTimestamp = Date.parse(end_timestamp) / 1000;
let res = await API.get(`/api/log/stat?type=${logType}&username=${username}&token_name=${token_name}&model_name=${model_name}&start_timestamp=${localStartTimestamp}&end_timestamp=${localEndTimestamp}&channel=${channel}`);
const {success, message, data} = res.data;
if (success) {
setStat(data);
} else {
showError(message);
}
};
const handleEyeClick = async () => {
setLoadingStat(true);
if (isAdminUser) {
await getLogStat();
} else {
await getLogSelfStat();
}
setShowStat(true);
setLoadingStat(false);
};
const showUserInfo = async (userId) => {
if (!isAdminUser) {
return;
}
const res = await API.get(`/api/user/${userId}`);
const {success, message, data} = res.data;
if (success) {
Modal.info({
title: '用户信息',
content: <div style={{padding: 12}}>
<p>用户名: {data.username}</p>
<p>余额: {renderQuota(data.quota)}</p>
<p>已用额度{renderQuota(data.used_quota)}</p>
<p>请求次数{renderNumber(data.request_count)}</p>
</div>,
centered: true,
})
} else {
showError(message);
}
};
const setLogsFormat = (logs) => {
for (let i = 0; i < logs.length; i++) {
logs[i].timestamp2string = timestamp2string(logs[i].created_at);
logs[i].key = '' + logs[i].id;
}
// data.key = '' + data.id
setLogs(logs);
setLogCount(logs.length + ITEMS_PER_PAGE);
// console.log(logCount);
} }
}, {
const loadLogs = async (startIdx, pageSize) => { title: '用户',
setLoading(true); dataIndex: 'username',
className: isAdmin() ? 'tableShow' : 'tableHiddle',
let url = ''; render: (text, record, index) => {
let localStartTimestamp = Date.parse(start_timestamp) / 1000; return (isAdminUser ? <div>
let localEndTimestamp = Date.parse(end_timestamp) / 1000; <Avatar size="small" color={stringToColor(text)} style={{ marginRight: 4 }}
if (isAdminUser) { onClick={() => showUserInfo(record.user_id)}>
url = `/api/log/?p=${startIdx}&page_size=${pageSize}&type=${logType}&username=${username}&token_name=${token_name}&model_name=${model_name}&start_timestamp=${localStartTimestamp}&end_timestamp=${localEndTimestamp}&channel=${channel}`; {typeof text === 'string' && text.slice(0, 1)}
} else { </Avatar>
url = `/api/log/self/?p=${startIdx}&page_size=${pageSize}&type=${logType}&token_name=${token_name}&model_name=${model_name}&start_timestamp=${localStartTimestamp}&end_timestamp=${localEndTimestamp}`; {text}
} </div> : <></>);
const res = await API.get(url);
const {success, message, data} = res.data;
if (success) {
if (startIdx === 0) {
setLogsFormat(data);
} else {
let newLogs = [...logs];
newLogs.splice(startIdx * pageSize, data.length, ...data);
setLogsFormat(newLogs);
}
} else {
showError(message);
}
setLoading(false);
};
const pageData = logs.slice((activePage - 1) * pageSize, activePage * pageSize);
const handlePageChange = page => {
setActivePage(page);
if (page === Math.ceil(logs.length / pageSize) + 1) {
// In this case we have to load more data and then append them.
loadLogs(page - 1, pageSize).then(r => {
});
}
};
const handlePageSizeChange = async (size) => {
localStorage.setItem('page-size', size + '')
setPageSize(size)
setActivePage(1)
loadLogs(0, size)
.then()
.catch((reason) => {
showError(reason);
})
};
const refresh = async () => {
// setLoading(true);
setActivePage(1);
await loadLogs(0, pageSize);
};
const copyText = async (text) => {
if (await copy(text)) {
showSuccess('已复制:' + text);
} else {
// setSearchKeyword(text);
Modal.error({title: '无法复制到剪贴板,请手动复制', content: text});
}
} }
}, {
title: '令牌', dataIndex: 'token_name', render: (text, record, index) => {
return (record.type === 0 || record.type === 2 ? <div>
<Tag color="grey" size="large" onClick={() => {
copyText(text);
}}> {text} </Tag>
</div> : <></>);
}
}, {
title: '类型', dataIndex: 'type', render: (text, record, index) => {
return (<div>
{renderType(text)}
</div>);
}
}, {
title: '模型', dataIndex: 'model_name', render: (text, record, index) => {
return (record.type === 0 || record.type === 2 ? <div>
<Tag color={stringToColor(text)} size="large" onClick={() => {
copyText(text);
}}> {text} </Tag>
</div> : <></>);
}
}, {
title: '用时', dataIndex: 'use_time', render: (text, record, index) => {
return (<div>
<Space>
{renderUseTime(text)}
{renderIsStream(record.is_stream)}
</Space>
</div>);
}
}, {
title: '提示', dataIndex: 'prompt_tokens', render: (text, record, index) => {
return (record.type === 0 || record.type === 2 ? <div>
{<span> {text} </span>}
</div> : <></>);
}
}, {
title: '补全', dataIndex: 'completion_tokens', render: (text, record, index) => {
return (parseInt(text) > 0 && (record.type === 0 || record.type === 2) ? <div>
{<span> {text} </span>}
</div> : <></>);
}
}, {
title: '花费', dataIndex: 'quota', render: (text, record, index) => {
return (record.type === 0 || record.type === 2 ? <div>
{renderQuota(text, 6)}
</div> : <></>);
}
}, {
title: '详情', dataIndex: 'content', render: (text, record, index) => {
return <Paragraph ellipsis={{ rows: 2, showTooltip: { type: 'popover', opts: { style: { width: 240 } } } }}
style={{ maxWidth: 240 }}>
{text}
</Paragraph>;
}
}];
// useEffect(() => { const [logs, setLogs] = useState([]);
// refresh().then(); const [showStat, setShowStat] = useState(false);
// }, [logType]); const [loading, setLoading] = useState(false);
const [loadingStat, setLoadingStat] = useState(false);
const [activePage, setActivePage] = useState(1);
const [logCount, setLogCount] = useState(ITEMS_PER_PAGE);
const [pageSize, setPageSize] = useState(ITEMS_PER_PAGE);
const [searchKeyword, setSearchKeyword] = useState('');
const [searching, setSearching] = useState(false);
const [logType, setLogType] = useState(0);
const isAdminUser = isAdmin();
let now = new Date();
// 初始化start_timestamp为前一天
const [inputs, setInputs] = useState({
username: '',
token_name: '',
model_name: '',
start_timestamp: timestamp2string(now.getTime() / 1000 - 86400),
end_timestamp: timestamp2string(now.getTime() / 1000 + 3600),
channel: ''
});
const { username, token_name, model_name, start_timestamp, end_timestamp, channel } = inputs;
useEffect(() => { const [stat, setStat] = useState({
// console.log('default effect') quota: 0, token: 0
const localPageSize = parseInt(localStorage.getItem('page-size')) || ITEMS_PER_PAGE; });
setPageSize(localPageSize)
loadLogs(0, localPageSize)
.then()
.catch((reason) => {
showError(reason);
});
}, []);
const searchLogs = async () => { const handleInputChange = (value, name) => {
if (searchKeyword === '') { setInputs((inputs) => ({ ...inputs, [name]: value }));
// if keyword is blank, load files instead. };
await loadLogs(0, pageSize);
setActivePage(1);
return;
}
setSearching(true);
const res = await API.get(`/api/log/self/search?keyword=${searchKeyword}`);
const {success, message, data} = res.data;
if (success) {
setLogs(data);
setActivePage(1);
} else {
showError(message);
}
setSearching(false);
};
return ( const getLogSelfStat = async () => {
let localStartTimestamp = Date.parse(start_timestamp) / 1000;
let localEndTimestamp = Date.parse(end_timestamp) / 1000;
let res = await API.get(`/api/log/self/stat?type=${logType}&token_name=${token_name}&model_name=${model_name}&start_timestamp=${localStartTimestamp}&end_timestamp=${localEndTimestamp}`);
const { success, message, data } = res.data;
if (success) {
setStat(data);
} else {
showError(message);
}
};
const getLogStat = async () => {
let localStartTimestamp = Date.parse(start_timestamp) / 1000;
let localEndTimestamp = Date.parse(end_timestamp) / 1000;
let res = await API.get(`/api/log/stat?type=${logType}&username=${username}&token_name=${token_name}&model_name=${model_name}&start_timestamp=${localStartTimestamp}&end_timestamp=${localEndTimestamp}&channel=${channel}`);
const { success, message, data } = res.data;
if (success) {
setStat(data);
} else {
showError(message);
}
};
const handleEyeClick = async () => {
setLoadingStat(true);
if (isAdminUser) {
await getLogStat();
} else {
await getLogSelfStat();
}
setShowStat(true);
setLoadingStat(false);
};
const showUserInfo = async (userId) => {
if (!isAdminUser) {
return;
}
const res = await API.get(`/api/user/${userId}`);
const { success, message, data } = res.data;
if (success) {
Modal.info({
title: '用户信息', content: <div style={{ padding: 12 }}>
<p>用户名: {data.username}</p>
<p>余额: {renderQuota(data.quota)}</p>
<p>已用额度{renderQuota(data.used_quota)}</p>
<p>请求次数{renderNumber(data.request_count)}</p>
</div>, centered: true
});
} else {
showError(message);
}
};
const setLogsFormat = (logs) => {
for (let i = 0; i < logs.length; i++) {
logs[i].timestamp2string = timestamp2string(logs[i].created_at);
logs[i].key = '' + logs[i].id;
}
// data.key = '' + data.id
setLogs(logs);
setLogCount(logs.length + ITEMS_PER_PAGE);
// console.log(logCount);
};
const loadLogs = async (startIdx, pageSize, logType = 0) => {
setLoading(true);
let url = '';
let localStartTimestamp = Date.parse(start_timestamp) / 1000;
let localEndTimestamp = Date.parse(end_timestamp) / 1000;
if (isAdminUser) {
url = `/api/log/?p=${startIdx}&page_size=${pageSize}&type=${logType}&username=${username}&token_name=${token_name}&model_name=${model_name}&start_timestamp=${localStartTimestamp}&end_timestamp=${localEndTimestamp}&channel=${channel}`;
} else {
url = `/api/log/self/?p=${startIdx}&page_size=${pageSize}&type=${logType}&token_name=${token_name}&model_name=${model_name}&start_timestamp=${localStartTimestamp}&end_timestamp=${localEndTimestamp}`;
}
const res = await API.get(url);
const { success, message, data } = res.data;
if (success) {
if (startIdx === 0) {
setLogsFormat(data);
} else {
let newLogs = [...logs];
newLogs.splice(startIdx * pageSize, data.length, ...data);
setLogsFormat(newLogs);
}
} else {
showError(message);
}
setLoading(false);
};
const pageData = logs.slice((activePage - 1) * pageSize, activePage * pageSize);
const handlePageChange = page => {
setActivePage(page);
if (page === Math.ceil(logs.length / pageSize) + 1) {
// In this case we have to load more data and then append them.
loadLogs(page - 1, pageSize).then(r => {
});
}
};
const handlePageSizeChange = async (size) => {
localStorage.setItem('page-size', size + '');
setPageSize(size);
setActivePage(1);
loadLogs(0, size)
.then()
.catch((reason) => {
showError(reason);
});
};
const refresh = async (localLogType) => {
// setLoading(true);
setActivePage(1);
await loadLogs(0, pageSize, localLogType);
};
const copyText = async (text) => {
if (await copy(text)) {
showSuccess('已复制:' + text);
} else {
// setSearchKeyword(text);
Modal.error({ title: '无法复制到剪贴板,请手动复制', content: text });
}
};
useEffect(() => {
// console.log('default effect')
const localPageSize = parseInt(localStorage.getItem('page-size')) || ITEMS_PER_PAGE;
setPageSize(localPageSize);
loadLogs(0, localPageSize)
.then()
.catch((reason) => {
showError(reason);
});
}, []);
const searchLogs = async () => {
if (searchKeyword === '') {
// if keyword is blank, load files instead.
await loadLogs(0, pageSize);
setActivePage(1);
return;
}
setSearching(true);
const res = await API.get(`/api/log/self/search?keyword=${searchKeyword}`);
const { success, message, data } = res.data;
if (success) {
setLogs(data);
setActivePage(1);
} else {
showError(message);
}
setSearching(false);
};
return (<>
<Layout>
<Header>
<Spin spinning={loadingStat}>
<h3>使用明细总消耗额度
<span onClick={handleEyeClick} style={{
cursor: 'pointer', color: 'gray'
}}>{showStat ? renderQuota(stat.quota) : '点击查看'}</span>
</h3>
</Spin>
</Header>
<Form layout="horizontal" style={{ marginTop: 10 }}>
<> <>
<Layout> <Form.Input field="token_name" label="令牌名称" style={{ width: 176 }} value={token_name}
<Header> placeholder={'可选值'} name="token_name"
<Spin spinning={loadingStat}> onChange={value => handleInputChange(value, 'token_name')} />
<h3>使用明细总消耗额度 <Form.Input field="model_name" label="模型名称" style={{ width: 176 }} value={model_name}
<span onClick={handleEyeClick} style={{ placeholder="可选值"
cursor: 'pointer', name="model_name"
color: 'gray' onChange={value => handleInputChange(value, 'model_name')} />
}}>{showStat ? renderQuota(stat.quota) : "点击查看"}</span> <Form.DatePicker field="start_timestamp" label="起始时间" style={{ width: 272 }}
initValue={start_timestamp}
</h3> value={start_timestamp} type="dateTime"
</Spin> name="start_timestamp"
</Header> onChange={value => handleInputChange(value, 'start_timestamp')} />
<Form layout='horizontal' style={{marginTop: 10}}> <Form.DatePicker field="end_timestamp" fluid label="结束时间" style={{ width: 272 }}
<> initValue={end_timestamp}
<Form.Input field="token_name" label='令牌名称' style={{width: 176}} value={token_name} value={end_timestamp} type="dateTime"
placeholder={'可选值'} name='token_name' name="end_timestamp"
onChange={value => handleInputChange(value, 'token_name')}/> onChange={value => handleInputChange(value, 'end_timestamp')} />
<Form.Input field="model_name" label='模型名称' style={{width: 176}} value={model_name} {isAdminUser && <>
placeholder='可选值' <Form.Input field="channel" label="渠道 ID" style={{ width: 176 }} value={channel}
name='model_name' placeholder="可选值" name="channel"
onChange={value => handleInputChange(value, 'model_name')}/> onChange={value => handleInputChange(value, 'channel')} />
<Form.DatePicker field="start_timestamp" label='起始时间' style={{width: 272}} <Form.Input field="username" label="用户名称" style={{ width: 176 }} value={username}
initValue={start_timestamp} placeholder={'可选值'} name="username"
value={start_timestamp} type='dateTime' onChange={value => handleInputChange(value, 'username')} />
name='start_timestamp' </>}
onChange={value => handleInputChange(value, 'start_timestamp')}/> <Form.Section>
<Form.DatePicker field="end_timestamp" fluid label='结束时间' style={{width: 272}} <Button label="查询" type="primary" htmlType="submit" className="btn-margin-right"
initValue={end_timestamp} onClick={refresh} loading={loading}>查询</Button>
value={end_timestamp} type='dateTime' </Form.Section>
name='end_timestamp'
onChange={value => handleInputChange(value, 'end_timestamp')}/>
{
isAdminUser && <>
<Form.Input field="channel" label='渠道 ID' style={{width: 176}} value={channel}
placeholder='可选值' name='channel'
onChange={value => handleInputChange(value, 'channel')}/>
<Form.Input field="username" label='用户名称' style={{width: 176}} value={username}
placeholder={'可选值'} name='username'
onChange={value => handleInputChange(value, 'username')}/>
</>
}
<Form.Section>
<Button label='查询' type="primary" htmlType="submit" className="btn-margin-right"
onClick={refresh} loading={loading}>查询</Button>
</Form.Section>
</>
</Form>
<Table style={{marginTop: 5}} columns={columns} dataSource={pageData} pagination={{
currentPage: activePage,
pageSize: pageSize,
total: logCount,
pageSizeOpts: [10, 20, 50, 100],
showSizeChanger: true,
onPageSizeChange: (size) => {
handlePageSizeChange(size).then()
},
onPageChange: handlePageChange,
}}/>
<Select defaultValue="0" style={{width: 120}} onChange={
(value) => {
setLogType(parseInt(value));
}
}>
<Select.Option value="0">全部</Select.Option>
<Select.Option value="1">充值</Select.Option>
<Select.Option value="2">消费</Select.Option>
<Select.Option value="3">管理</Select.Option>
<Select.Option value="4">系统</Select.Option>
</Select>
</Layout>
</> </>
); </Form>
<Table style={{ marginTop: 5 }} columns={columns} dataSource={pageData} pagination={{
currentPage: activePage,
pageSize: pageSize,
total: logCount,
pageSizeOpts: [10, 20, 50, 100],
showSizeChanger: true,
onPageSizeChange: (size) => {
handlePageSizeChange(size).then();
},
onPageChange: handlePageChange
}} />
<Select defaultValue="0" style={{ width: 120 }} onChange={(value) => {
setLogType(parseInt(value));
refresh(parseInt(value)).then();
}}>
<Select.Option value="0">全部</Select.Option>
<Select.Option value="1">充值</Select.Option>
<Select.Option value="2">消费</Select.Option>
<Select.Option value="3">管理</Select.Option>
<Select.Option value="4">系统</Select.Option>
</Select>
</Layout>
</>);
}; };
export default LogsTable; export default LogsTable;

View File

@ -1,454 +1,442 @@
import React, {useEffect, useState} from 'react'; import React, { useEffect, useState } from 'react';
import {API, copy, isAdmin, showError, showSuccess, timestamp2string} from '../helpers'; import { API, copy, isAdmin, showError, showSuccess, timestamp2string } from '../helpers';
import { import { Button, Form, ImagePreview, Layout, Modal, Progress, Table, Tag, Typography } from '@douyinfe/semi-ui';
Table, import { ITEMS_PER_PAGE } from '../constants';
Avatar,
Tag,
Form,
Button,
Layout,
Select,
Popover,
Modal,
ImagePreview,
Typography, Progress
} from '@douyinfe/semi-ui';
import {ITEMS_PER_PAGE} from '../constants';
import {renderNumber, renderQuota, stringToColor} from '../helpers/render';
const colors = ['amber', 'blue', 'cyan', 'green', 'grey', 'indigo', const colors = ['amber', 'blue', 'cyan', 'green', 'grey', 'indigo',
'light-blue', 'lime', 'orange', 'pink', 'light-blue', 'lime', 'orange', 'pink',
'purple', 'red', 'teal', 'violet', 'yellow' 'purple', 'red', 'teal', 'violet', 'yellow'
] ];
function renderType(type) { function renderType(type) {
switch (type) { switch (type) {
case 'IMAGINE': case 'IMAGINE':
return <Tag color="blue" size='large'>绘图</Tag>; return <Tag color="blue" size="large">绘图</Tag>;
case 'UPSCALE': case 'UPSCALE':
return <Tag color="orange" size='large'>放大</Tag>; return <Tag color="orange" size="large">放大</Tag>;
case 'VARIATION': case 'VARIATION':
return <Tag color="purple" size='large'>变换</Tag>; return <Tag color="purple" size="large">变换</Tag>;
case 'HIGH_VARIATION': case 'HIGH_VARIATION':
return <Tag color="purple" size='large'>强变换</Tag>; return <Tag color="purple" size="large">强变换</Tag>;
case 'LOW_VARIATION': case 'LOW_VARIATION':
return <Tag color="purple" size='large'>弱变换</Tag>; return <Tag color="purple" size="large">弱变换</Tag>;
case 'PAN': case 'PAN':
return <Tag color="cyan" size='large'>平移</Tag>; return <Tag color="cyan" size="large">平移</Tag>;
case 'DESCRIBE': case 'DESCRIBE':
return <Tag color="yellow" size='large'>图生文</Tag>; return <Tag color="yellow" size="large">图生文</Tag>;
case 'BLEND': case 'BLEND':
return <Tag color="lime" size='large'>图混合</Tag>; return <Tag color="lime" size="large">图混合</Tag>;
case 'SHORTEN': case 'SHORTEN':
return <Tag color="pink" size='large'>缩词</Tag>; return <Tag color="pink" size="large">缩词</Tag>;
case 'REROLL': case 'REROLL':
return <Tag color="indigo" size='large'>重绘</Tag>; return <Tag color="indigo" size="large">重绘</Tag>;
case 'INPAINT': case 'INPAINT':
return <Tag color="violet" size='large'>局部重绘-提交</Tag>; return <Tag color="violet" size="large">局部重绘-提交</Tag>;
case 'ZOOM': case 'ZOOM':
return <Tag color="teal" size='large'>变焦</Tag>; return <Tag color="teal" size="large">变焦</Tag>;
case 'CUSTOM_ZOOM': case 'CUSTOM_ZOOM':
return <Tag color="teal" size='large'>自定义变焦-提交</Tag>; return <Tag color="teal" size="large">自定义变焦-提交</Tag>;
case 'MODAL': case 'MODAL':
return <Tag color="green" size='large'>窗口处理</Tag>; return <Tag color="green" size="large">窗口处理</Tag>;
case 'SWAP_FACE': case 'SWAP_FACE':
return <Tag color="light-green" size='large'>换脸</Tag>; return <Tag color="light-green" size="large">换脸</Tag>;
default: default:
return <Tag color="white" size='large'>未知</Tag>; return <Tag color="white" size="large">未知</Tag>;
} }
} }
function renderCode(code) { function renderCode(code) {
switch (code) { switch (code) {
case 1: case 1:
return <Tag color="green" size='large'>已提交</Tag>; return <Tag color="green" size="large">已提交</Tag>;
case 21: case 21:
return <Tag color="lime" size='large'>等待中</Tag>; return <Tag color="lime" size="large">等待中</Tag>;
case 22: case 22:
return <Tag color="orange" size='large'>重复提交</Tag>; return <Tag color="orange" size="large">重复提交</Tag>;
case 0: case 0:
return <Tag color="yellow" size='large'>未提交</Tag>; return <Tag color="yellow" size="large">未提交</Tag>;
default: default:
return <Tag color="white" size='large'>未知</Tag>; return <Tag color="white" size="large">未知</Tag>;
} }
} }
function renderStatus(type) { function renderStatus(type) {
// Ensure all cases are string literals by adding quotes. // Ensure all cases are string literals by adding quotes.
switch (type) { switch (type) {
case 'SUCCESS': case 'SUCCESS':
return <Tag color="green" size='large'>成功</Tag>; return <Tag color="green" size="large">成功</Tag>;
case 'NOT_START': case 'NOT_START':
return <Tag color="grey" size='large'>未启动</Tag>; return <Tag color="grey" size="large">未启动</Tag>;
case 'SUBMITTED': case 'SUBMITTED':
return <Tag color="yellow" size='large'>队列中</Tag>; return <Tag color="yellow" size="large">队列中</Tag>;
case 'IN_PROGRESS': case 'IN_PROGRESS':
return <Tag color="blue" size='large'>执行中</Tag>; return <Tag color="blue" size="large">执行中</Tag>;
case 'FAILURE': case 'FAILURE':
return <Tag color="red" size='large'>失败</Tag>; return <Tag color="red" size="large">失败</Tag>;
case 'MODAL': case 'MODAL':
return <Tag color="yellow" size='large'>窗口等待</Tag>; return <Tag color="yellow" size="large">窗口等待</Tag>;
default: default:
return <Tag color="white" size='large'>未知</Tag>; return <Tag color="white" size="large">未知</Tag>;
} }
} }
const renderTimestamp = (timestampInSeconds) => { const renderTimestamp = (timestampInSeconds) => {
const date = new Date(timestampInSeconds * 1000); // 从秒转换为毫秒 const date = new Date(timestampInSeconds * 1000); // 从秒转换为毫秒
const year = date.getFullYear(); // 获取年份 const year = date.getFullYear(); // 获取年份
const month = ('0' + (date.getMonth() + 1)).slice(-2); // 获取月份从0开始需要+1并保证两位数 const month = ('0' + (date.getMonth() + 1)).slice(-2); // 获取月份从0开始需要+1并保证两位数
const day = ('0' + date.getDate()).slice(-2); // 获取日期,并保证两位数 const day = ('0' + date.getDate()).slice(-2); // 获取日期,并保证两位数
const hours = ('0' + date.getHours()).slice(-2); // 获取小时,并保证两位数 const hours = ('0' + date.getHours()).slice(-2); // 获取小时,并保证两位数
const minutes = ('0' + date.getMinutes()).slice(-2); // 获取分钟,并保证两位数 const minutes = ('0' + date.getMinutes()).slice(-2); // 获取分钟,并保证两位数
const seconds = ('0' + date.getSeconds()).slice(-2); // 获取秒钟,并保证两位数 const seconds = ('0' + date.getSeconds()).slice(-2); // 获取秒钟,并保证两位数
return `${year}-${month}-${day} ${hours}:${minutes}:${seconds}`; // 格式化输出 return `${year}-${month}-${day} ${hours}:${minutes}:${seconds}`; // 格式化输出
}; };
const LogsTable = () => { const LogsTable = () => {
const [isModalOpen, setIsModalOpen] = useState(false); const [isModalOpen, setIsModalOpen] = useState(false);
const [modalContent, setModalContent] = useState(''); const [modalContent, setModalContent] = useState('');
const columns = [ const columns = [
{ {
title: '提交时间', title: '提交时间',
dataIndex: 'submit_time', dataIndex: 'submit_time',
render: (text, record, index) => { render: (text, record, index) => {
return ( return (
<div> <div>
{renderTimestamp(text / 1000)} {renderTimestamp(text / 1000)}
</div> </div>
); );
}, }
}, },
{ {
title: '渠道', title: '渠道',
dataIndex: 'channel_id', dataIndex: 'channel_id',
className: isAdmin() ? 'tableShow' : 'tableHiddle', className: isAdmin() ? 'tableShow' : 'tableHiddle',
render: (text, record, index) => { render: (text, record, index) => {
return ( return (
<div> <div>
<Tag color={colors[parseInt(text) % colors.length]} size='large' onClick={() => { <Tag color={colors[parseInt(text) % colors.length]} size="large" onClick={() => {
copyText(text); // 假设copyText是用于文本复制的函数 copyText(text); // 假设copyText是用于文本复制的函数
}}> {text} </Tag> }}> {text} </Tag>
</div> </div>
); );
}, }
}, },
{ {
title: '类型', title: '类型',
dataIndex: 'action', dataIndex: 'action',
render: (text, record, index) => { render: (text, record, index) => {
return ( return (
<div> <div>
{renderType(text)} {renderType(text)}
</div> </div>
); );
}, }
}, },
{ {
title: '任务ID', title: '任务ID',
dataIndex: 'mj_id', dataIndex: 'mj_id',
render: (text, record, index) => { render: (text, record, index) => {
return ( return (
<div> <div>
{text} {text}
</div> </div>
); );
}, }
}, },
{ {
title: '提交结果', title: '提交结果',
dataIndex: 'code', dataIndex: 'code',
className: isAdmin() ? 'tableShow' : 'tableHiddle', className: isAdmin() ? 'tableShow' : 'tableHiddle',
render: (text, record, index) => { render: (text, record, index) => {
return ( return (
<div> <div>
{renderCode(text)} {renderCode(text)}
</div> </div>
); );
}, }
}, },
{ {
title: '任务状态', title: '任务状态',
dataIndex: 'status', dataIndex: 'status',
className: isAdmin() ? 'tableShow' : 'tableHiddle', className: isAdmin() ? 'tableShow' : 'tableHiddle',
render: (text, record, index) => { render: (text, record, index) => {
return ( return (
<div> <div>
{renderStatus(text)} {renderStatus(text)}
</div> </div>
); );
}, }
}, },
{ {
title: '进度', title: '进度',
dataIndex: 'progress', dataIndex: 'progress',
render: (text, record, index) => { render: (text, record, index) => {
return ( return (
<div> <div>
{ {
// 转换例如100%为数字100如果text未定义返回0 // 转换例如100%为数字100如果text未定义返回0
<Progress stroke={record.status === "FAILURE"?"var(--semi-color-warning)":null} percent={text ? parseInt(text.replace('%', '')) : 0} showInfo={true} <Progress stroke={record.status === 'FAILURE' ? 'var(--semi-color-warning)' : null}
aria-label="drawing progress"/> percent={text ? parseInt(text.replace('%', '')) : 0} showInfo={true}
} aria-label="drawing progress" />
</div>
);
},
},
{
title: '结果图片',
dataIndex: 'image_url',
render: (text, record, index) => {
if (!text) {
return '无';
}
return (
<Button
onClick={() => {
setModalImageUrl(text); // 更新图片URL状态
setIsModalOpenurl(true); // 打开模态框
}}
>
查看图片
</Button>
);
}
},
{
title: 'Prompt',
dataIndex: 'prompt',
render: (text, record, index) => {
// 如果text未定义返回替代文本例如空字符串''或其他
if (!text) {
return '无';
}
return (
<Typography.Text
ellipsis={{showTooltip: true}}
style={{width: 100}}
onClick={() => {
setModalContent(text);
setIsModalOpen(true);
}}
>
{text}
</Typography.Text>
);
}
},
{
title: 'PromptEn',
dataIndex: 'prompt_en',
render: (text, record, index) => {
// 如果text未定义返回替代文本例如空字符串''或其他
if (!text) {
return '无';
}
return (
<Typography.Text
ellipsis={{showTooltip: true}}
style={{width: 100}}
onClick={() => {
setModalContent(text);
setIsModalOpen(true);
}}
>
{text}
</Typography.Text>
);
}
},
{
title: '失败原因',
dataIndex: 'fail_reason',
render: (text, record, index) => {
// 如果text未定义返回替代文本例如空字符串''或其他
if (!text) {
return '无';
}
return (
<Typography.Text
ellipsis={{showTooltip: true}}
style={{width: 100}}
onClick={() => {
setModalContent(text);
setIsModalOpen(true);
}}
>
{text}
</Typography.Text>
);
} }
</div>
);
}
},
{
title: '结果图片',
dataIndex: 'image_url',
render: (text, record, index) => {
if (!text) {
return '无';
}
return (
<Button
onClick={() => {
setModalImageUrl(text); // 更新图片URL状态
setIsModalOpenurl(true); // 打开模态框
}}
>
查看图片
</Button>
);
}
},
{
title: 'Prompt',
dataIndex: 'prompt',
render: (text, record, index) => {
// 如果text未定义返回替代文本例如空字符串''或其他
if (!text) {
return '无';
} }
]; return (
<Typography.Text
const [logs, setLogs] = useState([]); ellipsis={{ showTooltip: true }}
const [loading, setLoading] = useState(true); style={{ width: 100 }}
const [activePage, setActivePage] = useState(1); onClick={() => {
const [logCount, setLogCount] = useState(ITEMS_PER_PAGE); setModalContent(text);
const [logType, setLogType] = useState(0); setIsModalOpen(true);
const isAdminUser = isAdmin(); }}
const [isModalOpenurl, setIsModalOpenurl] = useState(false); >
{text}
// 定义模态框图片URL的状态和更新函数 </Typography.Text>
const [modalImageUrl, setModalImageUrl] = useState(''); );
let now = new Date(); }
// 初始化start_timestamp为前一天 },
const [inputs, setInputs] = useState({ {
channel_id: '', title: 'PromptEn',
mj_id: '', dataIndex: 'prompt_en',
start_timestamp: timestamp2string(now.getTime() / 1000 - 2592000), render: (text, record, index) => {
end_timestamp: timestamp2string(now.getTime() / 1000 + 3600), // 如果text未定义返回替代文本例如空字符串''或其他
}); if (!text) {
const {channel_id, mj_id, start_timestamp, end_timestamp} = inputs; return '无';
const [stat, setStat] = useState({
quota: 0,
token: 0
});
const handleInputChange = (value, name) => {
setInputs((inputs) => ({...inputs, [name]: value}));
};
const setLogsFormat = (logs) => {
for (let i = 0; i < logs.length; i++) {
logs[i].timestamp2string = timestamp2string(logs[i].created_at);
logs[i].key = '' + logs[i].id;
} }
// data.key = '' + data.id
setLogs(logs); return (
setLogCount(logs.length + ITEMS_PER_PAGE); <Typography.Text
// console.log(logCount); ellipsis={{ showTooltip: true }}
style={{ width: 100 }}
onClick={() => {
setModalContent(text);
setIsModalOpen(true);
}}
>
{text}
</Typography.Text>
);
}
},
{
title: '失败原因',
dataIndex: 'fail_reason',
render: (text, record, index) => {
// 如果text未定义返回替代文本例如空字符串''或其他
if (!text) {
return '无';
}
return (
<Typography.Text
ellipsis={{ showTooltip: true }}
style={{ width: 100 }}
onClick={() => {
setModalContent(text);
setIsModalOpen(true);
}}
>
{text}
</Typography.Text>
);
}
} }
const loadLogs = async (startIdx) => { ];
setLoading(true);
let url = ''; const [logs, setLogs] = useState([]);
let localStartTimestamp = Date.parse(start_timestamp); const [loading, setLoading] = useState(true);
let localEndTimestamp = Date.parse(end_timestamp); const [activePage, setActivePage] = useState(1);
if (isAdminUser) { const [logCount, setLogCount] = useState(ITEMS_PER_PAGE);
url = `/api/mj/?p=${startIdx}&channel_id=${channel_id}&mj_id=${mj_id}&start_timestamp=${localStartTimestamp}&end_timestamp=${localEndTimestamp}`; const [logType, setLogType] = useState(0);
} else { const isAdminUser = isAdmin();
url = `/api/mj/self/?p=${startIdx}&mj_id=${mj_id}&start_timestamp=${localStartTimestamp}&end_timestamp=${localEndTimestamp}`; const [isModalOpenurl, setIsModalOpenurl] = useState(false);
}
const res = await API.get(url);
const {success, message, data} = res.data;
if (success) {
if (startIdx === 0) {
setLogsFormat(data);
} else {
let newLogs = [...logs];
newLogs.splice(startIdx * ITEMS_PER_PAGE, data.length, ...data);
setLogsFormat(newLogs);
}
} else {
showError(message);
}
setLoading(false);
};
const pageData = logs.slice((activePage - 1) * ITEMS_PER_PAGE, activePage * ITEMS_PER_PAGE); // 定义模态框图片URL的状态和更新函数
const [modalImageUrl, setModalImageUrl] = useState('');
let now = new Date();
// 初始化start_timestamp为前一天
const [inputs, setInputs] = useState({
channel_id: '',
mj_id: '',
start_timestamp: timestamp2string(now.getTime() / 1000 - 2592000),
end_timestamp: timestamp2string(now.getTime() / 1000 + 3600)
});
const { channel_id, mj_id, start_timestamp, end_timestamp } = inputs;
const handlePageChange = page => { const [stat, setStat] = useState({
setActivePage(page); quota: 0,
if (page === Math.ceil(logs.length / ITEMS_PER_PAGE) + 1) { token: 0
// In this case we have to load more data and then append them. });
loadLogs(page - 1).then(r => {
});
}
};
const refresh = async () => { const handleInputChange = (value, name) => {
// setLoading(true); setInputs((inputs) => ({ ...inputs, [name]: value }));
setActivePage(1); };
await loadLogs(0);
};
const copyText = async (text) => {
if (await copy(text)) { const setLogsFormat = (logs) => {
showSuccess('已复制:' + text); for (let i = 0; i < logs.length; i++) {
} else { logs[i].timestamp2string = timestamp2string(logs[i].created_at);
// setSearchKeyword(text); logs[i].key = '' + logs[i].id;
Modal.error({title: '无法复制到剪贴板,请手动复制', content: text});
}
} }
// data.key = '' + data.id
setLogs(logs);
setLogCount(logs.length + ITEMS_PER_PAGE);
// console.log(logCount);
};
useEffect(() => { const loadLogs = async (startIdx) => {
refresh().then(); setLoading(true);
}, [logType]);
let url = '';
let localStartTimestamp = Date.parse(start_timestamp);
let localEndTimestamp = Date.parse(end_timestamp);
if (isAdminUser) {
url = `/api/mj/?p=${startIdx}&channel_id=${channel_id}&mj_id=${mj_id}&start_timestamp=${localStartTimestamp}&end_timestamp=${localEndTimestamp}`;
} else {
url = `/api/mj/self/?p=${startIdx}&mj_id=${mj_id}&start_timestamp=${localStartTimestamp}&end_timestamp=${localEndTimestamp}`;
}
const res = await API.get(url);
const { success, message, data } = res.data;
if (success) {
if (startIdx === 0) {
setLogsFormat(data);
} else {
let newLogs = [...logs];
newLogs.splice(startIdx * ITEMS_PER_PAGE, data.length, ...data);
setLogsFormat(newLogs);
}
} else {
showError(message);
}
setLoading(false);
};
const pageData = logs.slice((activePage - 1) * ITEMS_PER_PAGE, activePage * ITEMS_PER_PAGE);
const handlePageChange = page => {
setActivePage(page);
if (page === Math.ceil(logs.length / ITEMS_PER_PAGE) + 1) {
// In this case we have to load more data and then append them.
loadLogs(page - 1).then(r => {
});
}
};
const refresh = async () => {
// setLoading(true);
setActivePage(1);
await loadLogs(0);
};
const copyText = async (text) => {
if (await copy(text)) {
showSuccess('已复制:' + text);
} else {
// setSearchKeyword(text);
Modal.error({ title: '无法复制到剪贴板,请手动复制', content: text });
}
};
useEffect(() => {
refresh().then();
}, [logType]);
return ( return (
<> <>
<Layout> <Layout>
<Form layout='horizontal' style={{marginTop: 10}}> <Form layout="horizontal" style={{ marginTop: 10 }}>
<> <>
<Form.Input field="channel_id" label='渠道 ID' style={{width: 176}} value={channel_id} <Form.Input field="channel_id" label="渠道 ID" style={{ width: 176 }} value={channel_id}
placeholder={'可选值'} name='channel_id' placeholder={'可选值'} name="channel_id"
onChange={value => handleInputChange(value, 'channel_id')}/> onChange={value => handleInputChange(value, 'channel_id')} />
<Form.Input field="mj_id" label='任务 ID' style={{width: 176}} value={mj_id} <Form.Input field="mj_id" label="任务 ID" style={{ width: 176 }} value={mj_id}
placeholder='可选值' placeholder="可选值"
name='mj_id' name="mj_id"
onChange={value => handleInputChange(value, 'mj_id')}/> onChange={value => handleInputChange(value, 'mj_id')} />
<Form.DatePicker field="start_timestamp" label='起始时间' style={{width: 272}} <Form.DatePicker field="start_timestamp" label="起始时间" style={{ width: 272 }}
initValue={start_timestamp} initValue={start_timestamp}
value={start_timestamp} type='dateTime' value={start_timestamp} type="dateTime"
name='start_timestamp' name="start_timestamp"
onChange={value => handleInputChange(value, 'start_timestamp')}/> onChange={value => handleInputChange(value, 'start_timestamp')} />
<Form.DatePicker field="end_timestamp" fluid label='结束时间' style={{width: 272}} <Form.DatePicker field="end_timestamp" fluid label="结束时间" style={{ width: 272 }}
initValue={end_timestamp} initValue={end_timestamp}
value={end_timestamp} type='dateTime' value={end_timestamp} type="dateTime"
name='end_timestamp' name="end_timestamp"
onChange={value => handleInputChange(value, 'end_timestamp')}/> onChange={value => handleInputChange(value, 'end_timestamp')} />
<Form.Section> <Form.Section>
<Button label='查询' type="primary" htmlType="submit" className="btn-margin-right" <Button label="查询" type="primary" htmlType="submit" className="btn-margin-right"
onClick={refresh}>查询</Button> onClick={refresh}>查询</Button>
</Form.Section> </Form.Section>
</> </>
</Form> </Form>
<Table style={{marginTop: 5}} columns={columns} dataSource={pageData} pagination={{ <Table style={{ marginTop: 5 }} columns={columns} dataSource={pageData} pagination={{
currentPage: activePage, currentPage: activePage,
pageSize: ITEMS_PER_PAGE, pageSize: ITEMS_PER_PAGE,
total: logCount, total: logCount,
pageSizeOpts: [10, 20, 50, 100], pageSizeOpts: [10, 20, 50, 100],
onPageChange: handlePageChange, onPageChange: handlePageChange
}} loading={loading}/> }} loading={loading} />
<Modal <Modal
visible={isModalOpen} visible={isModalOpen}
onOk={() => setIsModalOpen(false)} onOk={() => setIsModalOpen(false)}
onCancel={() => setIsModalOpen(false)} onCancel={() => setIsModalOpen(false)}
closable={null} closable={null}
bodyStyle={{height: '400px', overflow: 'auto'}} // 设置模态框内容区域样式 bodyStyle={{ height: '400px', overflow: 'auto' }} // 设置模态框内容区域样式
width={800} // 设置模态框宽度 width={800} // 设置模态框宽度
> >
<p style={{whiteSpace: 'pre-line'}}>{modalContent}</p> <p style={{ whiteSpace: 'pre-line' }}>{modalContent}</p>
</Modal> </Modal>
<ImagePreview <ImagePreview
src={modalImageUrl} src={modalImageUrl}
visible={isModalOpenurl} visible={isModalOpenurl}
onVisibleChange={(visible) => setIsModalOpenurl(visible)} onVisibleChange={(visible) => setIsModalOpenurl(visible)}
/> />
</Layout> </Layout>
</> </>
); );
}; };
export default LogsTable; export default LogsTable;

View File

@ -1,466 +1,466 @@
import React, {useEffect, useState} from 'react'; import React, { useEffect, useState } from 'react';
import {Divider, Form, Grid, Header} from 'semantic-ui-react'; import { Divider, Form, Grid, Header } from 'semantic-ui-react';
import {API, showError, showSuccess, timestamp2string, verifyJSON} from '../helpers'; import { API, showError, showSuccess, timestamp2string, verifyJSON } from '../helpers';
const OperationSetting = () => { const OperationSetting = () => {
let now = new Date(); let now = new Date();
let [inputs, setInputs] = useState({ let [inputs, setInputs] = useState({
QuotaForNewUser: 0, QuotaForNewUser: 0,
QuotaForInviter: 0, QuotaForInviter: 0,
QuotaForInvitee: 0, QuotaForInvitee: 0,
QuotaRemindThreshold: 0, QuotaRemindThreshold: 0,
PreConsumedQuota: 0, PreConsumedQuota: 0,
ModelRatio: '', ModelRatio: '',
ModelPrice: '', ModelPrice: '',
GroupRatio: '', GroupRatio: '',
TopUpLink: '', TopUpLink: '',
ChatLink: '', ChatLink: '',
ChatLink2: '', // 添加的新状态变量 ChatLink2: '', // 添加的新状态变量
QuotaPerUnit: 0, QuotaPerUnit: 0,
AutomaticDisableChannelEnabled: '', AutomaticDisableChannelEnabled: '',
AutomaticEnableChannelEnabled: '', AutomaticEnableChannelEnabled: '',
ChannelDisableThreshold: 0, ChannelDisableThreshold: 0,
LogConsumeEnabled: '', LogConsumeEnabled: '',
DisplayInCurrencyEnabled: '', DisplayInCurrencyEnabled: '',
DisplayTokenStatEnabled: '', DisplayTokenStatEnabled: '',
MjNotifyEnabled: '', MjNotifyEnabled: '',
DrawingEnabled: '', DrawingEnabled: '',
DataExportEnabled: '', DataExportEnabled: '',
DataExportDefaultTime: 'hour', DataExportDefaultTime: 'hour',
DataExportInterval: 5, DataExportInterval: 5,
DefaultCollapseSidebar: '', // 默认折叠侧边栏 DefaultCollapseSidebar: '', // 默认折叠侧边栏
RetryTimes: 0 RetryTimes: 0
});
const [originInputs, setOriginInputs] = useState({});
let [loading, setLoading] = useState(false);
let [historyTimestamp, setHistoryTimestamp] = useState(timestamp2string(now.getTime() / 1000 - 30 * 24 * 3600)); // a month ago
// 精确时间选项(小时,天,周)
const timeOptions = [
{ key: 'hour', text: '小时', value: 'hour' },
{ key: 'day', text: '天', value: 'day' },
{ key: 'week', text: '周', value: 'week' }
];
const getOptions = async () => {
const res = await API.get('/api/option/');
const { success, message, data } = res.data;
if (success) {
let newInputs = {};
data.forEach((item) => {
if (item.key === 'ModelRatio' || item.key === 'GroupRatio' || item.key === 'ModelPrice') {
item.value = JSON.stringify(JSON.parse(item.value), null, 2);
}
newInputs[item.key] = item.value;
});
setInputs(newInputs);
setOriginInputs(newInputs);
} else {
showError(message);
}
};
useEffect(() => {
getOptions().then();
}, []);
const updateOption = async (key, value) => {
setLoading(true);
if (key.endsWith('Enabled')) {
value = inputs[key] === 'true' ? 'false' : 'true';
}
if (key === 'DefaultCollapseSidebar') {
value = inputs[key] === 'true' ? 'false' : 'true';
}
console.log(key, value);
const res = await API.put('/api/option/', {
key,
value
}); });
const [originInputs, setOriginInputs] = useState({}); const { success, message } = res.data;
let [loading, setLoading] = useState(false); if (success) {
let [historyTimestamp, setHistoryTimestamp] = useState(timestamp2string(now.getTime() / 1000 - 30 * 24 * 3600)); // a month ago setInputs((inputs) => ({ ...inputs, [key]: value }));
// 精确时间选项(小时,天,周) } else {
const timeOptions = [ showError(message);
{key: 'hour', text: '小时', value: 'hour'}, }
{key: 'day', text: '天', value: 'day'}, setLoading(false);
{key: 'week', text: '周', value: 'week'} };
];
const getOptions = async () => {
const res = await API.get('/api/option/');
const {success, message, data} = res.data;
if (success) {
let newInputs = {};
data.forEach((item) => {
if (item.key === 'ModelRatio' || item.key === 'GroupRatio' || item.key === 'ModelPrice') {
item.value = JSON.stringify(JSON.parse(item.value), null, 2);
}
newInputs[item.key] = item.value;
});
setInputs(newInputs);
setOriginInputs(newInputs);
} else {
showError(message);
}
};
useEffect(() => { const handleInputChange = async (e, { name, value }) => {
getOptions().then(); if (name.endsWith('Enabled') || name === 'DataExportInterval' || name === 'DataExportDefaultTime' || name === 'DefaultCollapseSidebar') {
}, []); if (name === 'DataExportDefaultTime') {
localStorage.setItem('data_export_default_time', value);
}
await updateOption(name, value);
} else {
setInputs((inputs) => ({ ...inputs, [name]: value }));
}
};
const updateOption = async (key, value) => { const submitConfig = async (group) => {
setLoading(true); switch (group) {
if (key.endsWith('Enabled')) { case 'monitor':
value = inputs[key] === 'true' ? 'false' : 'true'; if (originInputs['ChannelDisableThreshold'] !== inputs.ChannelDisableThreshold) {
await updateOption('ChannelDisableThreshold', inputs.ChannelDisableThreshold);
} }
if (key === 'DefaultCollapseSidebar') { if (originInputs['QuotaRemindThreshold'] !== inputs.QuotaRemindThreshold) {
value = inputs[key] === 'true' ? 'false' : 'true'; await updateOption('QuotaRemindThreshold', inputs.QuotaRemindThreshold);
} }
console.log(key, value) break;
const res = await API.put('/api/option/', { case 'ratio':
key, if (originInputs['ModelRatio'] !== inputs.ModelRatio) {
value if (!verifyJSON(inputs.ModelRatio)) {
}); showError('模型倍率不是合法的 JSON 字符串');
const {success, message} = res.data;
if (success) {
setInputs((inputs) => ({...inputs, [key]: value}));
} else {
showError(message);
}
setLoading(false);
};
const handleInputChange = async (e, {name, value}) => {
if (name.endsWith('Enabled') || name === 'DataExportInterval' || name === 'DataExportDefaultTime' || name === 'DefaultCollapseSidebar') {
if (name === 'DataExportDefaultTime') {
localStorage.setItem('data_export_default_time', value);
}
await updateOption(name, value);
} else {
setInputs((inputs) => ({...inputs, [name]: value}));
}
};
const submitConfig = async (group) => {
switch (group) {
case 'monitor':
if (originInputs['ChannelDisableThreshold'] !== inputs.ChannelDisableThreshold) {
await updateOption('ChannelDisableThreshold', inputs.ChannelDisableThreshold);
}
if (originInputs['QuotaRemindThreshold'] !== inputs.QuotaRemindThreshold) {
await updateOption('QuotaRemindThreshold', inputs.QuotaRemindThreshold);
}
break;
case 'ratio':
if (originInputs['ModelRatio'] !== inputs.ModelRatio) {
if (!verifyJSON(inputs.ModelRatio)) {
showError('模型倍率不是合法的 JSON 字符串');
return;
}
await updateOption('ModelRatio', inputs.ModelRatio);
}
if (originInputs['GroupRatio'] !== inputs.GroupRatio) {
if (!verifyJSON(inputs.GroupRatio)) {
showError('分组倍率不是合法的 JSON 字符串');
return;
}
await updateOption('GroupRatio', inputs.GroupRatio);
}
if (originInputs['ModelPrice'] !== inputs.ModelPrice) {
if (!verifyJSON(inputs.ModelPrice)) {
showError('模型固定价格不是合法的 JSON 字符串');
return;
}
await updateOption('ModelPrice', inputs.ModelPrice);
}
break;
case 'quota':
if (originInputs['QuotaForNewUser'] !== inputs.QuotaForNewUser) {
await updateOption('QuotaForNewUser', inputs.QuotaForNewUser);
}
if (originInputs['QuotaForInvitee'] !== inputs.QuotaForInvitee) {
await updateOption('QuotaForInvitee', inputs.QuotaForInvitee);
}
if (originInputs['QuotaForInviter'] !== inputs.QuotaForInviter) {
await updateOption('QuotaForInviter', inputs.QuotaForInviter);
}
if (originInputs['PreConsumedQuota'] !== inputs.PreConsumedQuota) {
await updateOption('PreConsumedQuota', inputs.PreConsumedQuota);
}
break;
case 'general':
if (originInputs['TopUpLink'] !== inputs.TopUpLink) {
await updateOption('TopUpLink', inputs.TopUpLink);
}
if (originInputs['ChatLink'] !== inputs.ChatLink) {
await updateOption('ChatLink', inputs.ChatLink);
}
if (originInputs['ChatLink2'] !== inputs.ChatLink2) {
await updateOption('ChatLink2', inputs.ChatLink2);
}
if (originInputs['QuotaPerUnit'] !== inputs.QuotaPerUnit) {
await updateOption('QuotaPerUnit', inputs.QuotaPerUnit);
}
if (originInputs['RetryTimes'] !== inputs.RetryTimes) {
await updateOption('RetryTimes', inputs.RetryTimes);
}
break;
}
};
const deleteHistoryLogs = async () => {
console.log(inputs);
const res = await API.delete(`/api/log/?target_timestamp=${Date.parse(historyTimestamp) / 1000}`);
const {success, message, data} = res.data;
if (success) {
showSuccess(`${data} 条日志已清理!`);
return; return;
}
await updateOption('ModelRatio', inputs.ModelRatio);
} }
showError('日志清理失败:' + message); if (originInputs['GroupRatio'] !== inputs.GroupRatio) {
}; if (!verifyJSON(inputs.GroupRatio)) {
return ( showError('分组倍率不是合法的 JSON 字符串');
<Grid columns={1}> return;
<Grid.Column> }
<Form loading={loading}> await updateOption('GroupRatio', inputs.GroupRatio);
<Header as='h3'> }
通用设置 if (originInputs['ModelPrice'] !== inputs.ModelPrice) {
</Header> if (!verifyJSON(inputs.ModelPrice)) {
<Form.Group widths={4}> showError('模型固定价格不是合法的 JSON 字符串');
<Form.Input return;
label='充值链接' }
name='TopUpLink' await updateOption('ModelPrice', inputs.ModelPrice);
onChange={handleInputChange} }
autoComplete='new-password' break;
value={inputs.TopUpLink} case 'quota':
type='link' if (originInputs['QuotaForNewUser'] !== inputs.QuotaForNewUser) {
placeholder='例如发卡网站的购买链接' await updateOption('QuotaForNewUser', inputs.QuotaForNewUser);
/> }
<Form.Input if (originInputs['QuotaForInvitee'] !== inputs.QuotaForInvitee) {
label='默认聊天页面链接' await updateOption('QuotaForInvitee', inputs.QuotaForInvitee);
name='ChatLink' }
onChange={handleInputChange} if (originInputs['QuotaForInviter'] !== inputs.QuotaForInviter) {
autoComplete='new-password' await updateOption('QuotaForInviter', inputs.QuotaForInviter);
value={inputs.ChatLink} }
type='link' if (originInputs['PreConsumedQuota'] !== inputs.PreConsumedQuota) {
placeholder='例如 ChatGPT Next Web 的部署地址' await updateOption('PreConsumedQuota', inputs.PreConsumedQuota);
/> }
<Form.Input break;
label='聊天页面2链接' case 'general':
name='ChatLink2' if (originInputs['TopUpLink'] !== inputs.TopUpLink) {
onChange={handleInputChange} await updateOption('TopUpLink', inputs.TopUpLink);
autoComplete='new-password' }
value={inputs.ChatLink2} if (originInputs['ChatLink'] !== inputs.ChatLink) {
type='link' await updateOption('ChatLink', inputs.ChatLink);
placeholder='例如 ChatGPT Web & Midjourney 的部署地址' }
/> if (originInputs['ChatLink2'] !== inputs.ChatLink2) {
<Form.Input await updateOption('ChatLink2', inputs.ChatLink2);
label='单位美元额度' }
name='QuotaPerUnit' if (originInputs['QuotaPerUnit'] !== inputs.QuotaPerUnit) {
onChange={handleInputChange} await updateOption('QuotaPerUnit', inputs.QuotaPerUnit);
autoComplete='new-password' }
value={inputs.QuotaPerUnit} if (originInputs['RetryTimes'] !== inputs.RetryTimes) {
type='number' await updateOption('RetryTimes', inputs.RetryTimes);
step='0.01' }
placeholder='一单位货币能兑换的额度' break;
/> }
<Form.Input };
label='失败重试次数'
name='RetryTimes' const deleteHistoryLogs = async () => {
type={'number'} console.log(inputs);
step='1' const res = await API.delete(`/api/log/?target_timestamp=${Date.parse(historyTimestamp) / 1000}`);
min='0' const { success, message, data } = res.data;
onChange={handleInputChange} if (success) {
autoComplete='new-password' showSuccess(`${data} 条日志已清理!`);
value={inputs.RetryTimes} return;
placeholder='失败重试次数' }
/> showError('日志清理失败:' + message);
</Form.Group> };
<Form.Group inline> return (
<Form.Checkbox <Grid columns={1}>
checked={inputs.DisplayInCurrencyEnabled === 'true'} <Grid.Column>
label='以货币形式显示额度' <Form loading={loading}>
name='DisplayInCurrencyEnabled' <Header as="h3">
onChange={handleInputChange} 通用设置
/> </Header>
<Form.Checkbox <Form.Group widths={4}>
checked={inputs.DisplayTokenStatEnabled === 'true'} <Form.Input
label='Billing 相关 API 显示令牌额度而非用户额度' label="充值链接"
name='DisplayTokenStatEnabled' name="TopUpLink"
onChange={handleInputChange} onChange={handleInputChange}
/> autoComplete="new-password"
<Form.Checkbox value={inputs.TopUpLink}
checked={inputs.DefaultCollapseSidebar === 'true'} type="link"
label='默认折叠侧边栏' placeholder="例如发卡网站的购买链接"
name='DefaultCollapseSidebar' />
onChange={handleInputChange} <Form.Input
/> label="默认聊天页面链接"
</Form.Group> name="ChatLink"
<Form.Button onClick={() => { onChange={handleInputChange}
submitConfig('general').then(); autoComplete="new-password"
}}>保存通用设置</Form.Button> value={inputs.ChatLink}
<Divider/> type="link"
<Header as='h3'> placeholder="例如 ChatGPT Next Web 的部署地址"
绘图设置 />
</Header> <Form.Input
<Form.Group inline> label="聊天页面2链接"
<Form.Checkbox name="ChatLink2"
checked={inputs.DrawingEnabled === 'true'} onChange={handleInputChange}
label='启用绘图功能' autoComplete="new-password"
name='DrawingEnabled' value={inputs.ChatLink2}
onChange={handleInputChange} type="link"
/> placeholder="例如 ChatGPT Web & Midjourney 的部署地址"
<Form.Checkbox />
checked={inputs.MjNotifyEnabled === 'true'} <Form.Input
label='允许回调会泄露服务器ip地址' label="单位美元额度"
name='MjNotifyEnabled' name="QuotaPerUnit"
onChange={handleInputChange} onChange={handleInputChange}
/> autoComplete="new-password"
</Form.Group> value={inputs.QuotaPerUnit}
<Divider/> type="number"
<Header as='h3'> step="0.01"
日志设置 placeholder="一单位货币能兑换的额度"
</Header> />
<Form.Group inline> <Form.Input
<Form.Checkbox label="失败重试次数"
checked={inputs.LogConsumeEnabled === 'true'} name="RetryTimes"
label='启用额度消费日志记录' type={'number'}
name='LogConsumeEnabled' step="1"
onChange={handleInputChange} min="0"
/> onChange={handleInputChange}
</Form.Group> autoComplete="new-password"
<Form.Group widths={4}> value={inputs.RetryTimes}
<Form.Input label='目标时间' value={historyTimestamp} type='datetime-local' placeholder="失败重试次数"
name='history_timestamp' />
onChange={(e, {name, value}) => { </Form.Group>
setHistoryTimestamp(value); <Form.Group inline>
}}/> <Form.Checkbox
</Form.Group> checked={inputs.DisplayInCurrencyEnabled === 'true'}
<Form.Button onClick={() => { label="以货币形式显示额度"
deleteHistoryLogs().then(); name="DisplayInCurrencyEnabled"
}}>清理历史日志</Form.Button> onChange={handleInputChange}
<Divider/> />
<Header as='h3'> <Form.Checkbox
数据看板 checked={inputs.DisplayTokenStatEnabled === 'true'}
</Header> label="Billing 相关 API 显示令牌额度而非用户额度"
<Form.Checkbox name="DisplayTokenStatEnabled"
checked={inputs.DataExportEnabled === 'true'} onChange={handleInputChange}
label='启用数据看板(实验性)' />
name='DataExportEnabled' <Form.Checkbox
onChange={handleInputChange} checked={inputs.DefaultCollapseSidebar === 'true'}
/> label="默认折叠侧边栏"
<Form.Group> name="DefaultCollapseSidebar"
<Form.Input onChange={handleInputChange}
label='数据看板更新间隔(分钟,设置过短会影响数据库性能)' />
name='DataExportInterval' </Form.Group>
type={'number'} <Form.Button onClick={() => {
step='1' submitConfig('general').then();
min='1' }}>保存通用设置</Form.Button>
onChange={handleInputChange} <Divider />
autoComplete='new-password' <Header as="h3">
value={inputs.DataExportInterval} 绘图设置
placeholder='数据看板更新间隔(分钟,设置过短会影响数据库性能)' </Header>
/> <Form.Group inline>
<Form.Select <Form.Checkbox
label='数据看板默认时间粒度(仅修改展示粒度,统计精确到小时)' checked={inputs.DrawingEnabled === 'true'}
options={timeOptions} label="启用绘图功能"
name='DataExportDefaultTime' name="DrawingEnabled"
onChange={handleInputChange} onChange={handleInputChange}
autoComplete='new-password' />
value={inputs.DataExportDefaultTime} <Form.Checkbox
placeholder='数据看板默认时间粒度' checked={inputs.MjNotifyEnabled === 'true'}
/> label="允许回调会泄露服务器ip地址"
</Form.Group> name="MjNotifyEnabled"
<Divider/> onChange={handleInputChange}
<Header as='h3'> />
监控设置 </Form.Group>
</Header> <Divider />
<Form.Group widths={3}> <Header as="h3">
<Form.Input 日志设置
label='最长响应时间' </Header>
name='ChannelDisableThreshold' <Form.Group inline>
onChange={handleInputChange} <Form.Checkbox
autoComplete='new-password' checked={inputs.LogConsumeEnabled === 'true'}
value={inputs.ChannelDisableThreshold} label="启用额度消费日志记录"
type='number' name="LogConsumeEnabled"
min='0' onChange={handleInputChange}
placeholder='单位秒,当运行通道全部测试时,超过此时间将自动禁用通道' />
/> </Form.Group>
<Form.Input <Form.Group widths={4}>
label='额度提醒阈值' <Form.Input label="目标时间" value={historyTimestamp} type="datetime-local"
name='QuotaRemindThreshold' name="history_timestamp"
onChange={handleInputChange} onChange={(e, { name, value }) => {
autoComplete='new-password' setHistoryTimestamp(value);
value={inputs.QuotaRemindThreshold} }} />
type='number' </Form.Group>
min='0' <Form.Button onClick={() => {
placeholder='低于此额度时将发送邮件提醒用户' deleteHistoryLogs().then();
/> }}>清理历史日志</Form.Button>
</Form.Group> <Divider />
<Form.Group inline> <Header as="h3">
<Form.Checkbox 数据看板
checked={inputs.AutomaticDisableChannelEnabled === 'true'} </Header>
label='失败时自动禁用通道' <Form.Checkbox
name='AutomaticDisableChannelEnabled' checked={inputs.DataExportEnabled === 'true'}
onChange={handleInputChange} label="启用数据看板(实验性)"
/> name="DataExportEnabled"
<Form.Checkbox onChange={handleInputChange}
checked={inputs.AutomaticEnableChannelEnabled === 'true'} />
label='成功时自动启用通道' <Form.Group>
name='AutomaticEnableChannelEnabled' <Form.Input
onChange={handleInputChange} label="数据看板更新间隔(分钟,设置过短会影响数据库性能)"
/> name="DataExportInterval"
</Form.Group> type={'number'}
<Form.Button onClick={() => { step="1"
submitConfig('monitor').then(); min="1"
}}>保存监控设置</Form.Button> onChange={handleInputChange}
<Divider/> autoComplete="new-password"
<Header as='h3'> value={inputs.DataExportInterval}
额度设置 placeholder="数据看板更新间隔(分钟,设置过短会影响数据库性能)"
</Header> />
<Form.Group widths={4}> <Form.Select
<Form.Input label="数据看板默认时间粒度(仅修改展示粒度,统计精确到小时)"
label='新用户初始额度' options={timeOptions}
name='QuotaForNewUser' name="DataExportDefaultTime"
onChange={handleInputChange} onChange={handleInputChange}
autoComplete='new-password' autoComplete="new-password"
value={inputs.QuotaForNewUser} value={inputs.DataExportDefaultTime}
type='number' placeholder="数据看板默认时间粒度"
min='0' />
placeholder='例如100' </Form.Group>
/> <Divider />
<Form.Input <Header as="h3">
label='请求预扣费额度' 监控设置
name='PreConsumedQuota' </Header>
onChange={handleInputChange} <Form.Group widths={3}>
autoComplete='new-password' <Form.Input
value={inputs.PreConsumedQuota} label="最长响应时间"
type='number' name="ChannelDisableThreshold"
min='0' onChange={handleInputChange}
placeholder='请求结束后多退少补' autoComplete="new-password"
/> value={inputs.ChannelDisableThreshold}
<Form.Input type="number"
label='邀请新用户奖励额度' min="0"
name='QuotaForInviter' placeholder="单位秒,当运行通道全部测试时,超过此时间将自动禁用通道"
onChange={handleInputChange} />
autoComplete='new-password' <Form.Input
value={inputs.QuotaForInviter} label="额度提醒阈值"
type='number' name="QuotaRemindThreshold"
min='0' onChange={handleInputChange}
placeholder='例如2000' autoComplete="new-password"
/> value={inputs.QuotaRemindThreshold}
<Form.Input type="number"
label='新用户使用邀请码奖励额度' min="0"
name='QuotaForInvitee' placeholder="低于此额度时将发送邮件提醒用户"
onChange={handleInputChange} />
autoComplete='new-password' </Form.Group>
value={inputs.QuotaForInvitee} <Form.Group inline>
type='number' <Form.Checkbox
min='0' checked={inputs.AutomaticDisableChannelEnabled === 'true'}
placeholder='例如1000' label="失败时自动禁用通道"
/> name="AutomaticDisableChannelEnabled"
</Form.Group> onChange={handleInputChange}
<Form.Button onClick={() => { />
submitConfig('quota').then(); <Form.Checkbox
}}>保存额度设置</Form.Button> checked={inputs.AutomaticEnableChannelEnabled === 'true'}
<Divider/> label="成功时自动启用通道"
<Header as='h3'> name="AutomaticEnableChannelEnabled"
倍率设置 onChange={handleInputChange}
</Header> />
<Form.Group widths='equal'> </Form.Group>
<Form.TextArea <Form.Button onClick={() => {
label='模型固定价格(一次调用消耗多少刀,优先级大于模型倍率)' submitConfig('monitor').then();
name='ModelPrice' }}>保存监控设置</Form.Button>
onChange={handleInputChange} <Divider />
style={{minHeight: 250, fontFamily: 'JetBrains Mono, Consolas'}} <Header as="h3">
autoComplete='new-password' 额度设置
value={inputs.ModelPrice} </Header>
placeholder='为一个 JSON 文本,键为模型名称,值为一次调用消耗多少刀,比如 "gpt-4-gizmo-*": 0.1一次消耗0.1刀' <Form.Group widths={4}>
/> <Form.Input
</Form.Group> label="新用户初始额度"
<Form.Group widths='equal'> name="QuotaForNewUser"
<Form.TextArea onChange={handleInputChange}
label='模型倍率' autoComplete="new-password"
name='ModelRatio' value={inputs.QuotaForNewUser}
onChange={handleInputChange} type="number"
style={{minHeight: 250, fontFamily: 'JetBrains Mono, Consolas'}} min="0"
autoComplete='new-password' placeholder="例如100"
value={inputs.ModelRatio} />
placeholder='为一个 JSON 文本,键为模型名称,值为倍率' <Form.Input
/> label="请求预扣费额度"
</Form.Group> name="PreConsumedQuota"
<Form.Group widths='equal'> onChange={handleInputChange}
<Form.TextArea autoComplete="new-password"
label='分组倍率' value={inputs.PreConsumedQuota}
name='GroupRatio' type="number"
onChange={handleInputChange} min="0"
style={{minHeight: 250, fontFamily: 'JetBrains Mono, Consolas'}} placeholder="请求结束后多退少补"
autoComplete='new-password' />
value={inputs.GroupRatio} <Form.Input
placeholder='为一个 JSON 文本,键为分组名称,值为倍率' label="邀请新用户奖励额度"
/> name="QuotaForInviter"
</Form.Group> onChange={handleInputChange}
<Form.Button onClick={() => { autoComplete="new-password"
submitConfig('ratio').then(); value={inputs.QuotaForInviter}
}}>保存倍率设置</Form.Button> type="number"
</Form> min="0"
</Grid.Column> placeholder="例如2000"
</Grid> />
) <Form.Input
; label="新用户使用邀请码奖励额度"
name="QuotaForInvitee"
onChange={handleInputChange}
autoComplete="new-password"
value={inputs.QuotaForInvitee}
type="number"
min="0"
placeholder="例如1000"
/>
</Form.Group>
<Form.Button onClick={() => {
submitConfig('quota').then();
}}>保存额度设置</Form.Button>
<Divider />
<Header as="h3">
倍率设置
</Header>
<Form.Group widths="equal">
<Form.TextArea
label="模型固定价格(一次调用消耗多少刀,优先级大于模型倍率)"
name="ModelPrice"
onChange={handleInputChange}
style={{ minHeight: 250, fontFamily: 'JetBrains Mono, Consolas' }}
autoComplete="new-password"
value={inputs.ModelPrice}
placeholder='为一个 JSON 文本,键为模型名称,值为一次调用消耗多少刀,比如 "gpt-4-gizmo-*": 0.1一次消耗0.1刀'
/>
</Form.Group>
<Form.Group widths="equal">
<Form.TextArea
label="模型倍率"
name="ModelRatio"
onChange={handleInputChange}
style={{ minHeight: 250, fontFamily: 'JetBrains Mono, Consolas' }}
autoComplete="new-password"
value={inputs.ModelRatio}
placeholder="为一个 JSON 文本,键为模型名称,值为倍率"
/>
</Form.Group>
<Form.Group widths="equal">
<Form.TextArea
label="分组倍率"
name="GroupRatio"
onChange={handleInputChange}
style={{ minHeight: 250, fontFamily: 'JetBrains Mono, Consolas' }}
autoComplete="new-password"
value={inputs.GroupRatio}
placeholder="为一个 JSON 文本,键为分组名称,值为倍率"
/>
</Form.Group>
<Form.Button onClick={() => {
submitConfig('ratio').then();
}}>保存倍率设置</Form.Button>
</Form>
</Grid.Column>
</Grid>
)
;
}; };
export default OperationSetting; export default OperationSetting;

View File

@ -1,5 +1,5 @@
import React, { useEffect, useRef, useState } from 'react'; import React, { useEffect, useRef, useState } from 'react';
import { Col, Row , Form, Button, Banner } from '@douyinfe/semi-ui'; import { Banner, Button, Col, Form, Row } from '@douyinfe/semi-ui';
import { API, showError, showSuccess } from '../helpers'; import { API, showError, showSuccess } from '../helpers';
import { marked } from 'marked'; import { marked } from 'marked';
@ -57,8 +57,8 @@ const OtherSetting = () => {
await updateOption('Notice', inputs.Notice); await updateOption('Notice', inputs.Notice);
showSuccess('公告已更新'); showSuccess('公告已更新');
} catch (error) { } catch (error) {
console.error("公告更新失败", error); console.error('公告更新失败', error);
showError("公告更新失败") showError('公告更新失败');
} finally { } finally {
setLoadingInput((loadingInput) => ({ ...loadingInput, Notice: false })); setLoadingInput((loadingInput) => ({ ...loadingInput, Notice: false }));
} }
@ -72,8 +72,8 @@ const OtherSetting = () => {
await updateOption('SystemName', inputs.SystemName); await updateOption('SystemName', inputs.SystemName);
showSuccess('系统名称已更新'); showSuccess('系统名称已更新');
} catch (error) { } catch (error) {
console.error("系统名称更新失败", error); console.error('系统名称更新失败', error);
showError("系统名称更新失败") showError('系统名称更新失败');
} finally { } finally {
setLoadingInput((loadingInput) => ({ ...loadingInput, SystemName: false })); setLoadingInput((loadingInput) => ({ ...loadingInput, SystemName: false }));
} }
@ -86,8 +86,8 @@ const OtherSetting = () => {
await updateOption('Logo', inputs.Logo); await updateOption('Logo', inputs.Logo);
showSuccess('Logo 已更新'); showSuccess('Logo 已更新');
} catch (error) { } catch (error) {
console.error("Logo 更新失败", error); console.error('Logo 更新失败', error);
showError("Logo 更新失败") showError('Logo 更新失败');
} finally { } finally {
setLoadingInput((loadingInput) => ({ ...loadingInput, Logo: false })); setLoadingInput((loadingInput) => ({ ...loadingInput, Logo: false }));
} }
@ -99,8 +99,8 @@ const OtherSetting = () => {
await updateOption(key, inputs[key]); await updateOption(key, inputs[key]);
showSuccess('首页内容已更新'); showSuccess('首页内容已更新');
} catch (error) { } catch (error) {
console.error("首页内容更新失败", error); console.error('首页内容更新失败', error);
showError("首页内容更新失败") showError('首页内容更新失败');
} finally { } finally {
setLoadingInput((loadingInput) => ({ ...loadingInput, HomePageContent: false })); setLoadingInput((loadingInput) => ({ ...loadingInput, HomePageContent: false }));
} }
@ -112,8 +112,8 @@ const OtherSetting = () => {
await updateOption('About', inputs.About); await updateOption('About', inputs.About);
showSuccess('关于内容已更新'); showSuccess('关于内容已更新');
} catch (error) { } catch (error) {
console.error("关于内容更新失败", error); console.error('关于内容更新失败', error);
showError("关于内容更新失败"); showError('关于内容更新失败');
} finally { } finally {
setLoadingInput((loadingInput) => ({ ...loadingInput, About: false })); setLoadingInput((loadingInput) => ({ ...loadingInput, About: false }));
} }
@ -125,16 +125,14 @@ const OtherSetting = () => {
await updateOption('Footer', inputs.Footer); await updateOption('Footer', inputs.Footer);
showSuccess('页脚内容已更新'); showSuccess('页脚内容已更新');
} catch (error) { } catch (error) {
console.error("页脚内容更新失败", error); console.error('页脚内容更新失败', error);
showError("页脚内容更新失败"); showError('页脚内容更新失败');
} finally { } finally {
setLoadingInput((loadingInput) => ({ ...loadingInput, Footer: false })); setLoadingInput((loadingInput) => ({ ...loadingInput, Footer: false }));
} }
}; };
const openGitHubRelease = () => { const openGitHubRelease = () => {
window.location = window.location =
'https://github.com/songquanpeng/one-api/releases/latest'; 'https://github.com/songquanpeng/one-api/releases/latest';
@ -173,16 +171,17 @@ const OtherSetting = () => {
} }
}; };
useEffect( () => { useEffect(() => {
getOptions(); getOptions();
}, []); }, []);
return ( return (
<Row > <Row>
<Col span={24}> <Col span={24}>
{/* 通用设置 */} {/* 通用设置 */}
<Form values={inputs} getFormApi={formAPI => formAPISettingGeneral.current = formAPI} style={{marginBottom: 15}}> <Form values={inputs} getFormApi={formAPI => formAPISettingGeneral.current = formAPI}
style={{ marginBottom: 15 }}>
<Form.Section text={'通用设置'}> <Form.Section text={'通用设置'}>
<Form.TextArea <Form.TextArea
label={'公告'} label={'公告'}
@ -191,26 +190,27 @@ const OtherSetting = () => {
onChange={handleInputChange} onChange={handleInputChange}
style={{ fontFamily: 'JetBrains Mono, Consolas' }} style={{ fontFamily: 'JetBrains Mono, Consolas' }}
autosize={{ minRows: 6, maxRows: 12 }} autosize={{ minRows: 6, maxRows: 12 }}
/> />
<Button onClick={submitNotice} loading={loadingInput['Notice']}>设置公告</Button> <Button onClick={submitNotice} loading={loadingInput['Notice']}>设置公告</Button>
</Form.Section> </Form.Section>
</Form> </Form>
{/* 个性化设置 */} {/* 个性化设置 */}
<Form values={inputs} getFormApi={formAPI => formAPIPersonalization.current = formAPI} style={{marginBottom: 15}}> <Form values={inputs} getFormApi={formAPI => formAPIPersonalization.current = formAPI}
style={{ marginBottom: 15 }}>
<Form.Section text={'个性化设置'}> <Form.Section text={'个性化设置'}>
<Form.Input <Form.Input
label={'系统名称'} label={'系统名称'}
placeholder={'在此输入系统名称'} placeholder={'在此输入系统名称'}
field={'SystemName'} field={'SystemName'}
onChange={handleInputChange} onChange={handleInputChange}
/> />
<Button onClick={submitSystemName} loading={loadingInput['SystemName']}>设置系统名称</Button> <Button onClick={submitSystemName} loading={loadingInput['SystemName']}>设置系统名称</Button>
<Form.Input <Form.Input
label={'Logo 图片地址'} label={'Logo 图片地址'}
placeholder={'在此输入 Logo 图片地址'} placeholder={'在此输入 Logo 图片地址'}
field={'Logo'} field={'Logo'}
onChange={handleInputChange} onChange={handleInputChange}
/> />
<Button onClick={submitLogo} loading={loadingInput['Logo']}>设置 Logo</Button> <Button onClick={submitLogo} loading={loadingInput['Logo']}>设置 Logo</Button>
<Form.TextArea <Form.TextArea
label={'首页内容'} label={'首页内容'}
@ -219,8 +219,9 @@ const OtherSetting = () => {
onChange={handleInputChange} onChange={handleInputChange}
style={{ fontFamily: 'JetBrains Mono, Consolas' }} style={{ fontFamily: 'JetBrains Mono, Consolas' }}
autosize={{ minRows: 6, maxRows: 12 }} autosize={{ minRows: 6, maxRows: 12 }}
/> />
<Button onClick={() => submitOption('HomePageContent')} loading={loadingInput['HomePageContent']}>设置首页内容</Button> <Button onClick={() => submitOption('HomePageContent')}
loading={loadingInput['HomePageContent']}>设置首页内容</Button>
<Form.TextArea <Form.TextArea
label={'关于'} label={'关于'}
placeholder={'在此输入新的关于内容,支持 Markdown & HTML 代码。如果输入的是一个链接,则会使用该链接作为 iframe 的 src 属性,这允许你设置任意网页作为关于页面。'} placeholder={'在此输入新的关于内容,支持 Markdown & HTML 代码。如果输入的是一个链接,则会使用该链接作为 iframe 的 src 属性,这允许你设置任意网页作为关于页面。'}
@ -228,7 +229,7 @@ const OtherSetting = () => {
onChange={handleInputChange} onChange={handleInputChange}
style={{ fontFamily: 'JetBrains Mono, Consolas' }} style={{ fontFamily: 'JetBrains Mono, Consolas' }}
autosize={{ minRows: 6, maxRows: 12 }} autosize={{ minRows: 6, maxRows: 12 }}
/> />
<Button onClick={submitAbout} loading={loadingInput['About']}>设置关于</Button> <Button onClick={submitAbout} loading={loadingInput['About']}>设置关于</Button>
{/* */} {/* */}
<Banner <Banner
@ -236,14 +237,14 @@ const OtherSetting = () => {
type="info" type="info"
description="移除 One API 的版权标识必须首先获得授权,项目维护需要花费大量精力,如果本项目对你有意义,请主动支持本项目。" description="移除 One API 的版权标识必须首先获得授权,项目维护需要花费大量精力,如果本项目对你有意义,请主动支持本项目。"
closeIcon={null} closeIcon={null}
style={{ marginTop: 15 }} style={{ marginTop: 15 }}
/> />
<Form.Input <Form.Input
label={'页脚'} label={'页脚'}
placeholder={'在此输入新的页脚,留空则使用默认页脚,支持 HTML 代码'} placeholder={'在此输入新的页脚,留空则使用默认页脚,支持 HTML 代码'}
field={'Footer'} field={'Footer'}
onChange={handleInputChange} onChange={handleInputChange}
/> />
<Button onClick={submitFooter} loading={loadingInput['Footer']}>设置页脚</Button> <Button onClick={submitFooter} loading={loadingInput['Footer']}>设置页脚</Button>
</Form.Section> </Form.Section>
</Form> </Form>
@ -270,7 +271,7 @@ const OtherSetting = () => {
{/* />*/} {/* />*/}
{/* </Modal.Actions>*/} {/* </Modal.Actions>*/}
{/*</Modal>*/} {/*</Modal>*/}
</Row> </Row>
); );
}; };

View File

@ -1,12 +1,12 @@
import React, { useEffect, useState } from 'react'; import React, { useEffect, useState } from 'react';
import { Button, Form, Grid, Header, Image, Segment } from 'semantic-ui-react'; import { Button, Form, Grid, Header, Image, Segment } from 'semantic-ui-react';
import { API, copy, showError, showInfo, showNotice, showSuccess } from '../helpers'; import { API, copy, showError, showNotice } from '../helpers';
import { useSearchParams } from 'react-router-dom'; import { useSearchParams } from 'react-router-dom';
const PasswordResetConfirm = () => { const PasswordResetConfirm = () => {
const [inputs, setInputs] = useState({ const [inputs, setInputs] = useState({
email: '', email: '',
token: '', token: ''
}); });
const { email, token } = inputs; const { email, token } = inputs;
@ -23,7 +23,7 @@ const PasswordResetConfirm = () => {
let email = searchParams.get('email'); let email = searchParams.get('email');
setInputs({ setInputs({
token, token,
email, email
}); });
}, []); }, []);
@ -37,7 +37,7 @@ const PasswordResetConfirm = () => {
setDisableButton(false); setDisableButton(false);
setCountdown(30); setCountdown(30);
} }
return () => clearInterval(countdownInterval); return () => clearInterval(countdownInterval);
}, [disableButton, countdown]); }, [disableButton, countdown]);
async function handleSubmit(e) { async function handleSubmit(e) {
@ -46,7 +46,7 @@ const PasswordResetConfirm = () => {
setLoading(true); setLoading(true);
const res = await API.post(`/api/user/reset`, { const res = await API.post(`/api/user/reset`, {
email, email,
token, token
}); });
const { success, message } = res.data; const { success, message } = res.data;
if (success) { if (success) {
@ -59,44 +59,44 @@ const PasswordResetConfirm = () => {
} }
setLoading(false); setLoading(false);
} }
return ( return (
<Grid textAlign='center' style={{ marginTop: '48px' }}> <Grid textAlign="center" style={{ marginTop: '48px' }}>
<Grid.Column style={{ maxWidth: 450 }}> <Grid.Column style={{ maxWidth: 450 }}>
<Header as='h2' color='' textAlign='center'> <Header as="h2" color="" textAlign="center">
<Image src='/logo.png' /> 密码重置确认 <Image src="/logo.png" /> 密码重置确认
</Header> </Header>
<Form size='large'> <Form size="large">
<Segment> <Segment>
<Form.Input <Form.Input
fluid fluid
icon='mail' icon="mail"
iconPosition='left' iconPosition="left"
placeholder='邮箱地址' placeholder="邮箱地址"
name='email' name="email"
value={email} value={email}
readOnly readOnly
/> />
{newPassword && ( {newPassword && (
<Form.Input <Form.Input
fluid fluid
icon='lock' icon="lock"
iconPosition='left' iconPosition="left"
placeholder='新密码' placeholder="新密码"
name='newPassword' name="newPassword"
value={newPassword} value={newPassword}
readOnly readOnly
onClick={(e) => { onClick={(e) => {
e.target.select(); e.target.select();
navigator.clipboard.writeText(newPassword); navigator.clipboard.writeText(newPassword);
showNotice(`密码已复制到剪贴板:${newPassword}`); showNotice(`密码已复制到剪贴板:${newPassword}`);
}} }}
/> />
)} )}
<Button <Button
color='green' color="green"
fluid fluid
size='large' size="large"
onClick={handleSubmit} onClick={handleSubmit}
loading={loading} loading={loading}
disabled={disableButton} disabled={disableButton}
@ -107,7 +107,7 @@ const PasswordResetConfirm = () => {
</Form> </Form>
</Grid.Column> </Grid.Column>
</Grid> </Grid>
); );
}; };
export default PasswordResetConfirm; export default PasswordResetConfirm;

View File

@ -56,19 +56,19 @@ const PasswordResetForm = () => {
} }
return ( return (
<Grid textAlign='center' style={{ marginTop: '48px' }}> <Grid textAlign="center" style={{ marginTop: '48px' }}>
<Grid.Column style={{ maxWidth: 450 }}> <Grid.Column style={{ maxWidth: 450 }}>
<Header as='h2' color='' textAlign='center'> <Header as="h2" color="" textAlign="center">
<Image src='/logo.png' /> 密码重置 <Image src="/logo.png" /> 密码重置
</Header> </Header>
<Form size='large'> <Form size="large">
<Segment> <Segment>
<Form.Input <Form.Input
fluid fluid
icon='mail' icon="mail"
iconPosition='left' iconPosition="left"
placeholder='邮箱地址' placeholder="邮箱地址"
name='email' name="email"
value={email} value={email}
onChange={handleChange} onChange={handleChange}
/> />
@ -83,9 +83,9 @@ const PasswordResetForm = () => {
<></> <></>
)} )}
<Button <Button
color='green' color="green"
fluid fluid
size='large' size="large"
onClick={handleSubmit} onClick={handleSubmit}
loading={loading} loading={loading}
disabled={disableButton} disabled={disableButton}

File diff suppressed because it is too large Load Diff

View File

@ -5,7 +5,7 @@ import { history } from '../helpers';
function PrivateRoute({ children }) { function PrivateRoute({ children }) {
if (!localStorage.getItem('user')) { if (!localStorage.getItem('user')) {
return <Navigate to='/login' state={{ from: history.location }} />; return <Navigate to="/login" state={{ from: history.location }} />;
} }
return children; return children;
} }

View File

@ -1,406 +1,406 @@
import React, {useEffect, useState} from 'react'; import React, { useEffect, useState } from 'react';
import {API, copy, showError, showInfo, showSuccess, showWarning, timestamp2string} from '../helpers'; import { API, copy, showError, showSuccess, timestamp2string } from '../helpers';
import {ITEMS_PER_PAGE} from '../constants'; import { ITEMS_PER_PAGE } from '../constants';
import {renderQuota} from '../helpers/render'; import { renderQuota } from '../helpers/render';
import {Button, Modal, Popconfirm, Popover, Table, Tag, Form} from "@douyinfe/semi-ui"; import { Button, Form, Modal, Popconfirm, Popover, Table, Tag } from '@douyinfe/semi-ui';
import EditRedemption from "../pages/Redemption/EditRedemption"; import EditRedemption from '../pages/Redemption/EditRedemption';
function renderTimestamp(timestamp) { function renderTimestamp(timestamp) {
return ( return (
<> <>
{timestamp2string(timestamp)} {timestamp2string(timestamp)}
</> </>
); );
} }
function renderStatus(status) { function renderStatus(status) {
switch (status) { switch (status) {
case 1: case 1:
return <Tag color='green' size='large'>未使用</Tag>; return <Tag color="green" size="large">未使用</Tag>;
case 2: case 2:
return <Tag color='red' size='large'> 已禁用 </Tag>; return <Tag color="red" size="large"> 已禁用 </Tag>;
case 3: case 3:
return <Tag color='grey' size='large'> 已使用 </Tag>; return <Tag color="grey" size="large"> 已使用 </Tag>;
default: default:
return <Tag color='black' size='large'> 未知状态 </Tag>; return <Tag color="black" size="large"> 未知状态 </Tag>;
} }
} }
const RedemptionsTable = () => { const RedemptionsTable = () => {
const columns = [ const columns = [
{ {
title: 'ID', title: 'ID',
dataIndex: 'id', dataIndex: 'id'
}, },
{ {
title: '名称', title: '名称',
dataIndex: 'name', dataIndex: 'name'
}, },
{ {
title: '状态', title: '状态',
dataIndex: 'status', dataIndex: 'status',
key: 'status', key: 'status',
render: (text, record, index) => { render: (text, record, index) => {
return ( return (
<div> <div>
{renderStatus(text)} {renderStatus(text)}
</div> </div>
); );
}, }
}, },
{ {
title: '额度', title: '额度',
dataIndex: 'quota', dataIndex: 'quota',
render: (text, record, index) => { render: (text, record, index) => {
return ( return (
<div> <div>
{renderQuota(parseInt(text))} {renderQuota(parseInt(text))}
</div> </div>
); );
}, }
}, },
{ {
title: '创建时间', title: '创建时间',
dataIndex: 'created_time', dataIndex: 'created_time',
render: (text, record, index) => { render: (text, record, index) => {
return ( return (
<div> <div>
{renderTimestamp(text)} {renderTimestamp(text)}
</div> </div>
); );
}, }
}, },
{ {
title: '兑换人ID', title: '兑换人ID',
dataIndex: 'used_user_id', dataIndex: 'used_user_id',
render: (text, record, index) => { render: (text, record, index) => {
return ( return (
<div> <div>
{text === 0 ? '无' : text} {text === 0 ? '无' : text}
</div> </div>
); );
}, }
}, },
{ {
title: '', title: '',
dataIndex: 'operate', dataIndex: 'operate',
render: (text, record, index) => ( render: (text, record, index) => (
<div> <div>
<Popover <Popover
content={ content={
record.key record.key
}
style={{padding: 20}}
position="top"
>
<Button theme='light' type='tertiary' style={{marginRight: 1}}>查看</Button>
</Popover>
<Button theme='light' type='secondary' style={{marginRight: 1}}
onClick={async (text) => {
await copyText(record.key)
}}
>复制</Button>
<Popconfirm
title="确定是否要删除此兑换码?"
content="此修改将不可逆"
okType={'danger'}
position={'left'}
onConfirm={() => {
manageRedemption(record.id, 'delete', record).then(
() => {
removeRecord(record.key);
}
)
}}
>
<Button theme='light' type='danger' style={{marginRight: 1}}>删除</Button>
</Popconfirm>
{
record.status === 1 ?
<Button theme='light' type='warning' style={{marginRight: 1}} onClick={
async () => {
manageRedemption(
record.id,
'disable',
record
)
}
}>禁用</Button> :
<Button theme='light' type='secondary' style={{marginRight: 1}} onClick={
async () => {
manageRedemption(
record.id,
'enable',
record
);
}
} disabled={record.status === 3}>启用</Button>
}
<Button theme='light' type='tertiary' style={{marginRight: 1}} onClick={
() => {
setEditingRedemption(record);
setShowEdit(true);
}
} disabled={record.status !== 1}>编辑</Button>
</div>
),
},
];
const [redemptions, setRedemptions] = useState([]);
const [loading, setLoading] = useState(true);
const [activePage, setActivePage] = useState(1);
const [searchKeyword, setSearchKeyword] = useState('');
const [searching, setSearching] = useState(false);
const [tokenCount, setTokenCount] = useState(ITEMS_PER_PAGE);
const [selectedKeys, setSelectedKeys] = useState([]);
const [editingRedemption, setEditingRedemption] = useState({
id: undefined,
});
const [showEdit, setShowEdit] = useState(false);
const closeEdit = () => {
setShowEdit(false);
}
// const setCount = (data) => {
// if (data.length >= (activePage) * ITEMS_PER_PAGE) {
// setTokenCount(data.length + 1);
// } else {
// setTokenCount(data.length);
// }
// }
const setRedemptionFormat = (redeptions) => {
// for (let i = 0; i < redeptions.length; i++) {
// redeptions[i].key = '' + redeptions[i].id;
// }
// data.key = '' + data.id
setRedemptions(redeptions);
if (redeptions.length >= (activePage) * ITEMS_PER_PAGE) {
setTokenCount(redeptions.length + 1);
} else {
setTokenCount(redeptions.length);
}
}
const loadRedemptions = async (startIdx) => {
const res = await API.get(`/api/redemption/?p=${startIdx}`);
const {success, message, data} = res.data;
if (success) {
if (startIdx === 0) {
setRedemptionFormat(data);
} else {
let newRedemptions = redemptions;
newRedemptions.push(...data);
setRedemptionFormat(newRedemptions);
} }
} else { style={{ padding: 20 }}
showError(message); position="top"
} >
setLoading(false); <Button theme="light" type="tertiary" style={{ marginRight: 1 }}>查看</Button>
}; </Popover>
<Button theme="light" type="secondary" style={{ marginRight: 1 }}
const removeRecord = key => { onClick={async (text) => {
let newDataSource = [...redemptions]; await copyText(record.key);
if (key != null) { }}
let idx = newDataSource.findIndex(data => data.key === key); >复制</Button>
<Popconfirm
if (idx > -1) { title="确定是否要删除此兑换码?"
newDataSource.splice(idx, 1); content="此修改将不可逆"
setRedemptions(newDataSource); okType={'danger'}
} position={'left'}
} onConfirm={() => {
}; manageRedemption(record.id, 'delete', record).then(
const copyText = async (text) => {
if (await copy(text)) {
showSuccess('已复制到剪贴板!');
} else {
// setSearchKeyword(text);
Modal.error({title: '无法复制到剪贴板,请手动复制', content: text});
}
}
const onPaginationChange = (e, {activePage}) => {
(async () => {
if (activePage === Math.ceil(redemptions.length / ITEMS_PER_PAGE) + 1) {
// In this case we have to load more data and then append them.
await loadRedemptions(activePage - 1);
}
setActivePage(activePage);
})();
};
useEffect(() => {
loadRedemptions(0)
.then()
.catch((reason) => {
showError(reason);
});
}, []);
const refresh = async () => {
await loadRedemptions(activePage - 1);
};
const manageRedemption = async (id, action, record) => {
let data = {id};
let res;
switch (action) {
case 'delete':
res = await API.delete(`/api/redemption/${id}/`);
break;
case 'enable':
data.status = 1;
res = await API.put('/api/redemption/?status_only=true', data);
break;
case 'disable':
data.status = 2;
res = await API.put('/api/redemption/?status_only=true', data);
break;
}
const {success, message} = res.data;
if (success) {
showSuccess('操作成功完成!');
let redemption = res.data.data;
let newRedemptions = [...redemptions];
// let realIdx = (activePage - 1) * ITEMS_PER_PAGE + idx;
if (action === 'delete') {
} else {
record.status = redemption.status;
}
setRedemptions(newRedemptions);
} else {
showError(message);
}
};
const searchRedemptions = async () => {
if (searchKeyword === '') {
// if keyword is blank, load files instead.
await loadRedemptions(0);
setActivePage(1);
return;
}
setSearching(true);
const res = await API.get(`/api/redemption/search?keyword=${searchKeyword}`);
const {success, message, data} = res.data;
if (success) {
setRedemptions(data);
setActivePage(1);
} else {
showError(message);
}
setSearching(false);
};
const handleKeywordChange = async (value) => {
setSearchKeyword(value.trim());
};
const sortRedemption = (key) => {
if (redemptions.length === 0) return;
setLoading(true);
let sortedRedemptions = [...redemptions];
sortedRedemptions.sort((a, b) => {
return ('' + a[key]).localeCompare(b[key]);
});
if (sortedRedemptions[0].id === redemptions[0].id) {
sortedRedemptions.reverse();
}
setRedemptions(sortedRedemptions);
setLoading(false);
};
const handlePageChange = page => {
setActivePage(page);
if (page === Math.ceil(redemptions.length / ITEMS_PER_PAGE) + 1) {
// In this case we have to load more data and then append them.
loadRedemptions(page - 1).then(r => {
});
}
};
let pageData = redemptions.slice((activePage - 1) * ITEMS_PER_PAGE, activePage * ITEMS_PER_PAGE);
const rowSelection = {
onSelect: (record, selected) => {
},
onSelectAll: (selected, selectedRows) => {
},
onChange: (selectedRowKeys, selectedRows) => {
setSelectedKeys(selectedRows);
},
};
const handleRow = (record, index) => {
if (record.status !== 1) {
return {
style: {
background: 'var(--semi-color-disabled-border)',
},
};
} else {
return {};
}
};
return (
<>
<EditRedemption refresh={refresh} editingRedemption={editingRedemption} visiable={showEdit}
handleClose={closeEdit}></EditRedemption>
<Form onSubmit={searchRedemptions}>
<Form.Input
label='搜索关键字'
field='keyword'
icon='search'
iconPosition='left'
placeholder='关键字(id或者名称)'
value={searchKeyword}
loading={searching}
onChange={handleKeywordChange}
/>
</Form>
<Table style={{marginTop: 20}} columns={columns} dataSource={pageData} pagination={{
currentPage: activePage,
pageSize: ITEMS_PER_PAGE,
total: tokenCount,
// showSizeChanger: true,
// pageSizeOptions: [10, 20, 50, 100],
formatPageText: (page) => `${page.currentStart} - ${page.currentEnd} 条,共 ${redemptions.length}`,
// onPageSizeChange: (size) => {
// setPageSize(size);
// setActivePage(1);
// },
onPageChange: handlePageChange,
}} loading={loading} rowSelection={rowSelection} onRow={handleRow}>
</Table>
<Button theme='light' type='primary' style={{marginRight: 8}} onClick={
() => { () => {
setEditingRedemption({ removeRecord(record.key);
id: undefined,
});
setShowEdit(true);
} }
}>添加兑换码</Button> );
<Button label='复制所选兑换码' type="warning" onClick={ }}
>
<Button theme="light" type="danger" style={{ marginRight: 1 }}>删除</Button>
</Popconfirm>
{
record.status === 1 ?
<Button theme="light" type="warning" style={{ marginRight: 1 }} onClick={
async () => { async () => {
if (selectedKeys.length === 0) { manageRedemption(
showError('请至少选择一个兑换码!'); record.id,
return; 'disable',
} record
let keys = ""; );
for (let i = 0; i < selectedKeys.length; i++) {
keys += selectedKeys[i].name + " " + selectedKeys[i].key + "\n";
}
await copyText(keys);
} }
}>复制所选兑换码到剪贴板</Button> }>禁用</Button> :
</> <Button theme="light" type="secondary" style={{ marginRight: 1 }} onClick={
); async () => {
manageRedemption(
record.id,
'enable',
record
);
}
} disabled={record.status === 3}>启用</Button>
}
<Button theme="light" type="tertiary" style={{ marginRight: 1 }} onClick={
() => {
setEditingRedemption(record);
setShowEdit(true);
}
} disabled={record.status !== 1}>编辑</Button>
</div>
)
}
];
const [redemptions, setRedemptions] = useState([]);
const [loading, setLoading] = useState(true);
const [activePage, setActivePage] = useState(1);
const [searchKeyword, setSearchKeyword] = useState('');
const [searching, setSearching] = useState(false);
const [tokenCount, setTokenCount] = useState(ITEMS_PER_PAGE);
const [selectedKeys, setSelectedKeys] = useState([]);
const [editingRedemption, setEditingRedemption] = useState({
id: undefined
});
const [showEdit, setShowEdit] = useState(false);
const closeEdit = () => {
setShowEdit(false);
};
// const setCount = (data) => {
// if (data.length >= (activePage) * ITEMS_PER_PAGE) {
// setTokenCount(data.length + 1);
// } else {
// setTokenCount(data.length);
// }
// }
const setRedemptionFormat = (redeptions) => {
// for (let i = 0; i < redeptions.length; i++) {
// redeptions[i].key = '' + redeptions[i].id;
// }
// data.key = '' + data.id
setRedemptions(redeptions);
if (redeptions.length >= (activePage) * ITEMS_PER_PAGE) {
setTokenCount(redeptions.length + 1);
} else {
setTokenCount(redeptions.length);
}
};
const loadRedemptions = async (startIdx) => {
const res = await API.get(`/api/redemption/?p=${startIdx}`);
const { success, message, data } = res.data;
if (success) {
if (startIdx === 0) {
setRedemptionFormat(data);
} else {
let newRedemptions = redemptions;
newRedemptions.push(...data);
setRedemptionFormat(newRedemptions);
}
} else {
showError(message);
}
setLoading(false);
};
const removeRecord = key => {
let newDataSource = [...redemptions];
if (key != null) {
let idx = newDataSource.findIndex(data => data.key === key);
if (idx > -1) {
newDataSource.splice(idx, 1);
setRedemptions(newDataSource);
}
}
};
const copyText = async (text) => {
if (await copy(text)) {
showSuccess('已复制到剪贴板!');
} else {
// setSearchKeyword(text);
Modal.error({ title: '无法复制到剪贴板,请手动复制', content: text });
}
};
const onPaginationChange = (e, { activePage }) => {
(async () => {
if (activePage === Math.ceil(redemptions.length / ITEMS_PER_PAGE) + 1) {
// In this case we have to load more data and then append them.
await loadRedemptions(activePage - 1);
}
setActivePage(activePage);
})();
};
useEffect(() => {
loadRedemptions(0)
.then()
.catch((reason) => {
showError(reason);
});
}, []);
const refresh = async () => {
await loadRedemptions(activePage - 1);
};
const manageRedemption = async (id, action, record) => {
let data = { id };
let res;
switch (action) {
case 'delete':
res = await API.delete(`/api/redemption/${id}/`);
break;
case 'enable':
data.status = 1;
res = await API.put('/api/redemption/?status_only=true', data);
break;
case 'disable':
data.status = 2;
res = await API.put('/api/redemption/?status_only=true', data);
break;
}
const { success, message } = res.data;
if (success) {
showSuccess('操作成功完成!');
let redemption = res.data.data;
let newRedemptions = [...redemptions];
// let realIdx = (activePage - 1) * ITEMS_PER_PAGE + idx;
if (action === 'delete') {
} else {
record.status = redemption.status;
}
setRedemptions(newRedemptions);
} else {
showError(message);
}
};
const searchRedemptions = async () => {
if (searchKeyword === '') {
// if keyword is blank, load files instead.
await loadRedemptions(0);
setActivePage(1);
return;
}
setSearching(true);
const res = await API.get(`/api/redemption/search?keyword=${searchKeyword}`);
const { success, message, data } = res.data;
if (success) {
setRedemptions(data);
setActivePage(1);
} else {
showError(message);
}
setSearching(false);
};
const handleKeywordChange = async (value) => {
setSearchKeyword(value.trim());
};
const sortRedemption = (key) => {
if (redemptions.length === 0) return;
setLoading(true);
let sortedRedemptions = [...redemptions];
sortedRedemptions.sort((a, b) => {
return ('' + a[key]).localeCompare(b[key]);
});
if (sortedRedemptions[0].id === redemptions[0].id) {
sortedRedemptions.reverse();
}
setRedemptions(sortedRedemptions);
setLoading(false);
};
const handlePageChange = page => {
setActivePage(page);
if (page === Math.ceil(redemptions.length / ITEMS_PER_PAGE) + 1) {
// In this case we have to load more data and then append them.
loadRedemptions(page - 1).then(r => {
});
}
};
let pageData = redemptions.slice((activePage - 1) * ITEMS_PER_PAGE, activePage * ITEMS_PER_PAGE);
const rowSelection = {
onSelect: (record, selected) => {
},
onSelectAll: (selected, selectedRows) => {
},
onChange: (selectedRowKeys, selectedRows) => {
setSelectedKeys(selectedRows);
}
};
const handleRow = (record, index) => {
if (record.status !== 1) {
return {
style: {
background: 'var(--semi-color-disabled-border)'
}
};
} else {
return {};
}
};
return (
<>
<EditRedemption refresh={refresh} editingRedemption={editingRedemption} visiable={showEdit}
handleClose={closeEdit}></EditRedemption>
<Form onSubmit={searchRedemptions}>
<Form.Input
label="搜索关键字"
field="keyword"
icon="search"
iconPosition="left"
placeholder="关键字(id或者名称)"
value={searchKeyword}
loading={searching}
onChange={handleKeywordChange}
/>
</Form>
<Table style={{ marginTop: 20 }} columns={columns} dataSource={pageData} pagination={{
currentPage: activePage,
pageSize: ITEMS_PER_PAGE,
total: tokenCount,
// showSizeChanger: true,
// pageSizeOptions: [10, 20, 50, 100],
formatPageText: (page) => `${page.currentStart} - ${page.currentEnd} 条,共 ${redemptions.length}`,
// onPageSizeChange: (size) => {
// setPageSize(size);
// setActivePage(1);
// },
onPageChange: handlePageChange
}} loading={loading} rowSelection={rowSelection} onRow={handleRow}>
</Table>
<Button theme="light" type="primary" style={{ marginRight: 8 }} onClick={
() => {
setEditingRedemption({
id: undefined
});
setShowEdit(true);
}
}>添加兑换码</Button>
<Button label="复制所选兑换码" type="warning" onClick={
async () => {
if (selectedKeys.length === 0) {
showError('请至少选择一个兑换码!');
return;
}
let keys = '';
for (let i = 0; i < selectedKeys.length; i++) {
keys += selectedKeys[i].name + ' ' + selectedKeys[i].key + '\n';
}
await copyText(keys);
}
}>复制所选兑换码到剪贴板</Button>
</>
);
}; };
export default RedemptionsTable; export default RedemptionsTable;

View File

@ -98,49 +98,49 @@ const RegisterForm = () => {
}; };
return ( return (
<Grid textAlign='center' style={{ marginTop: '48px' }}> <Grid textAlign="center" style={{ marginTop: '48px' }}>
<Grid.Column style={{ maxWidth: 450 }}> <Grid.Column style={{ maxWidth: 450 }}>
<Header as='h2' color='' textAlign='center'> <Header as="h2" color="" textAlign="center">
<Image src={logo} /> 新用户注册 <Image src={logo} /> 新用户注册
</Header> </Header>
<Form size='large'> <Form size="large">
<Segment> <Segment>
<Form.Input <Form.Input
fluid fluid
icon='user' icon="user"
iconPosition='left' iconPosition="left"
placeholder='输入用户名,最长 12 位' placeholder="输入用户名,最长 12 位"
onChange={handleChange} onChange={handleChange}
name='username' name="username"
/> />
<Form.Input <Form.Input
fluid fluid
icon='lock' icon="lock"
iconPosition='left' iconPosition="left"
placeholder='输入密码,最短 8 位,最长 20 位' placeholder="输入密码,最短 8 位,最长 20 位"
onChange={handleChange} onChange={handleChange}
name='password' name="password"
type='password' type="password"
/> />
<Form.Input <Form.Input
fluid fluid
icon='lock' icon="lock"
iconPosition='left' iconPosition="left"
placeholder='输入密码,最短 8 位,最长 20 位' placeholder="输入密码,最短 8 位,最长 20 位"
onChange={handleChange} onChange={handleChange}
name='password2' name="password2"
type='password' type="password"
/> />
{showEmailVerification ? ( {showEmailVerification ? (
<> <>
<Form.Input <Form.Input
fluid fluid
icon='mail' icon="mail"
iconPosition='left' iconPosition="left"
placeholder='输入邮箱地址' placeholder="输入邮箱地址"
onChange={handleChange} onChange={handleChange}
name='email' name="email"
type='email' type="email"
action={ action={
<Button onClick={sendVerificationCode} disabled={loading}> <Button onClick={sendVerificationCode} disabled={loading}>
获取验证码 获取验证码
@ -149,11 +149,11 @@ const RegisterForm = () => {
/> />
<Form.Input <Form.Input
fluid fluid
icon='lock' icon="lock"
iconPosition='left' iconPosition="left"
placeholder='输入验证码' placeholder="输入验证码"
onChange={handleChange} onChange={handleChange}
name='verification_code' name="verification_code"
/> />
</> </>
) : ( ) : (
@ -170,9 +170,9 @@ const RegisterForm = () => {
<></> <></>
)} )}
<Button <Button
color='green' color="green"
fluid fluid
size='large' size="large"
onClick={handleSubmit} onClick={handleSubmit}
loading={loading} loading={loading}
> >
@ -182,7 +182,7 @@ const RegisterForm = () => {
</Form> </Form>
<Message> <Message>
已有账户 已有账户
<Link to='/login' className='btn btn-link'> <Link to="/login" className="btn btn-link">
点击登录 点击登录
</Link> </Link>
</Message> </Message>

View File

@ -1,213 +1,213 @@
import React, { useContext, useEffect, useLayoutEffect, useMemo, useState } from 'react'; import React, { useContext, useEffect, useMemo, useState } from 'react';
import {Link, useNavigate} from 'react-router-dom'; import { Link, useNavigate } from 'react-router-dom';
import {UserContext} from '../context/User'; import { UserContext } from '../context/User';
import { StatusContext } from '../context/Status'; import { StatusContext } from '../context/Status';
import { API, getLogo, getSystemName, isAdmin, isMobile, showError, showSuccess } from '../helpers'; import { API, getLogo, getSystemName, isAdmin, isMobile, showError } from '../helpers';
import '../index.css'; import '../index.css';
import { import {
IconCalendarClock, IconCalendarClock,
IconHistogram, IconComment,
IconGift, IconCreditCard,
IconKey, IconGift,
IconUser, IconHistogram,
IconLayers, IconHome,
IconSetting, IconImage,
IconCreditCard, IconKey,
IconComment, IconLayers,
IconHome, IconSetting,
IconImage IconUser
} from '@douyinfe/semi-icons'; } from '@douyinfe/semi-icons';
import {Nav, Avatar, Dropdown, Layout} from '@douyinfe/semi-ui'; import { Layout, Nav } from '@douyinfe/semi-ui';
// HeaderBar Buttons // HeaderBar Buttons
const SiderBar = () => { const SiderBar = () => {
const [userState, userDispatch] = useContext(UserContext); const [userState, userDispatch] = useContext(UserContext);
const [statusState, statusDispatch] = useContext(StatusContext); const [statusState, statusDispatch] = useContext(StatusContext);
const defaultIsCollapsed = isMobile() || localStorage.getItem('default_collapse_sidebar') === 'true'; const defaultIsCollapsed = isMobile() || localStorage.getItem('default_collapse_sidebar') === 'true';
let navigate = useNavigate(); let navigate = useNavigate();
const [selectedKeys, setSelectedKeys] = useState(['home']); const [selectedKeys, setSelectedKeys] = useState(['home']);
const systemName = getSystemName(); const systemName = getSystemName();
const logo = getLogo(); const logo = getLogo();
const [isCollapsed, setIsCollapsed] = useState(defaultIsCollapsed); const [isCollapsed, setIsCollapsed] = useState(defaultIsCollapsed);
const headerButtons = useMemo(() => [ const headerButtons = useMemo(() => [
{ {
text: '首页', text: '首页',
itemKey: 'home', itemKey: 'home',
to: '/', to: '/',
icon: <IconHome/> icon: <IconHome />
}, },
{ {
text: '渠道', text: '渠道',
itemKey: 'channel', itemKey: 'channel',
to: '/channel', to: '/channel',
icon: <IconLayers/>, icon: <IconLayers />,
className: isAdmin()?'semi-navigation-item-normal':'tableHiddle', className: isAdmin() ? 'semi-navigation-item-normal' : 'tableHiddle'
}, },
{ {
text: '聊天', text: '聊天',
itemKey: 'chat', itemKey: 'chat',
to: '/chat', to: '/chat',
icon: <IconComment />, icon: <IconComment />,
className: localStorage.getItem('chat_link')?'semi-navigation-item-normal':'tableHiddle', className: localStorage.getItem('chat_link') ? 'semi-navigation-item-normal' : 'tableHiddle'
}, },
{ {
text: '令牌', text: '令牌',
itemKey: 'token', itemKey: 'token',
to: '/token', to: '/token',
icon: <IconKey/> icon: <IconKey />
}, },
{ {
text: '兑换码', text: '兑换码',
itemKey: 'redemption', itemKey: 'redemption',
to: '/redemption', to: '/redemption',
icon: <IconGift/>, icon: <IconGift />,
className: isAdmin()?'semi-navigation-item-normal':'tableHiddle', className: isAdmin() ? 'semi-navigation-item-normal' : 'tableHiddle'
}, },
{ {
text: '钱包', text: '钱包',
itemKey: 'topup', itemKey: 'topup',
to: '/topup', to: '/topup',
icon: <IconCreditCard/> icon: <IconCreditCard />
}, },
{ {
text: '用户管理', text: '用户管理',
itemKey: 'user', itemKey: 'user',
to: '/user', to: '/user',
icon: <IconUser/>, icon: <IconUser />,
className: isAdmin()?'semi-navigation-item-normal':'tableHiddle', className: isAdmin() ? 'semi-navigation-item-normal' : 'tableHiddle'
}, },
{ {
text: '日志', text: '日志',
itemKey: 'log', itemKey: 'log',
to: '/log', to: '/log',
icon: <IconHistogram/> icon: <IconHistogram />
}, },
{ {
text: '数据看板', text: '数据看板',
itemKey: 'detail', itemKey: 'detail',
to: '/detail', to: '/detail',
icon: <IconCalendarClock />, icon: <IconCalendarClock />,
className: localStorage.getItem('enable_data_export') === 'true'?'semi-navigation-item-normal':'tableHiddle', className: localStorage.getItem('enable_data_export') === 'true' ? 'semi-navigation-item-normal' : 'tableHiddle'
}, },
{ {
text: '绘图', text: '绘图',
itemKey: 'midjourney', itemKey: 'midjourney',
to: '/midjourney', to: '/midjourney',
icon: <IconImage/>, icon: <IconImage />,
className: localStorage.getItem('enable_drawing') === 'true'?'semi-navigation-item-normal':'tableHiddle', className: localStorage.getItem('enable_drawing') === 'true' ? 'semi-navigation-item-normal' : 'tableHiddle'
}, },
{ {
text: '设置', text: '设置',
itemKey: 'setting', itemKey: 'setting',
to: '/setting', to: '/setting',
icon: <IconSetting/> icon: <IconSetting />
}, }
// { // {
// text: '关于', // text: '关于',
// itemKey: 'about', // itemKey: 'about',
// to: '/about', // to: '/about',
// icon: <IconAt/> // icon: <IconAt/>
// } // }
], [localStorage.getItem('enable_data_export'), localStorage.getItem('enable_drawing'), localStorage.getItem('chat_link'), isAdmin()]); ], [localStorage.getItem('enable_data_export'), localStorage.getItem('enable_drawing'), localStorage.getItem('chat_link'), isAdmin()]);
const loadStatus = async () => { const loadStatus = async () => {
const res = await API.get('/api/status'); const res = await API.get('/api/status');
const { success, data } = res.data; const { success, data } = res.data;
if (success) { if (success) {
localStorage.setItem('status', JSON.stringify(data)); localStorage.setItem('status', JSON.stringify(data));
statusDispatch({ type: 'set', payload: data }); statusDispatch({ type: 'set', payload: data });
localStorage.setItem('system_name', data.system_name); localStorage.setItem('system_name', data.system_name);
localStorage.setItem('logo', data.logo); localStorage.setItem('logo', data.logo);
localStorage.setItem('footer_html', data.footer_html); localStorage.setItem('footer_html', data.footer_html);
localStorage.setItem('quota_per_unit', data.quota_per_unit); localStorage.setItem('quota_per_unit', data.quota_per_unit);
localStorage.setItem('display_in_currency', data.display_in_currency); localStorage.setItem('display_in_currency', data.display_in_currency);
localStorage.setItem('enable_drawing', data.enable_drawing); localStorage.setItem('enable_drawing', data.enable_drawing);
localStorage.setItem('enable_data_export', data.enable_data_export); localStorage.setItem('enable_data_export', data.enable_data_export);
localStorage.setItem('data_export_default_time', data.data_export_default_time); localStorage.setItem('data_export_default_time', data.data_export_default_time);
localStorage.setItem('default_collapse_sidebar', data.default_collapse_sidebar); localStorage.setItem('default_collapse_sidebar', data.default_collapse_sidebar);
if (data.chat_link) { if (data.chat_link) {
localStorage.setItem('chat_link', data.chat_link); localStorage.setItem('chat_link', data.chat_link);
} else { } else {
localStorage.removeItem('chat_link'); localStorage.removeItem('chat_link');
} }
if (data.chat_link2) { if (data.chat_link2) {
localStorage.setItem('chat_link2', data.chat_link2); localStorage.setItem('chat_link2', data.chat_link2);
} else { } else {
localStorage.removeItem('chat_link2'); localStorage.removeItem('chat_link2');
} }
} else { } else {
showError('无法正常连接至服务器!'); showError('无法正常连接至服务器!');
} }
}; };
useEffect(() => { useEffect(() => {
loadStatus().then(() => { loadStatus().then(() => {
setIsCollapsed(isMobile() || localStorage.getItem('default_collapse_sidebar') === 'true'); setIsCollapsed(isMobile() || localStorage.getItem('default_collapse_sidebar') === 'true');
}); });
},[]) }, []);
return ( return (
<> <>
<Layout> <Layout>
<div style={{height: '100%'}}> <div style={{ height: '100%' }}>
<Nav <Nav
// bodyStyle={{ maxWidth: 200 }} // bodyStyle={{ maxWidth: 200 }}
style={{ maxWidth: 200 }} style={{ maxWidth: 200 }}
defaultIsCollapsed={isMobile() || localStorage.getItem('default_collapse_sidebar') === 'true'} defaultIsCollapsed={isMobile() || localStorage.getItem('default_collapse_sidebar') === 'true'}
isCollapsed={isCollapsed} isCollapsed={isCollapsed}
onCollapseChange={collapsed => { onCollapseChange={collapsed => {
setIsCollapsed(collapsed); setIsCollapsed(collapsed);
}} }}
selectedKeys={selectedKeys} selectedKeys={selectedKeys}
renderWrapper={({itemElement, isSubNav, isInSubNav, props}) => { renderWrapper={({ itemElement, isSubNav, isInSubNav, props }) => {
const routerMap = { const routerMap = {
home: "/", home: '/',
channel: "/channel", channel: '/channel',
token: "/token", token: '/token',
redemption: "/redemption", redemption: '/redemption',
topup: "/topup", topup: '/topup',
user: "/user", user: '/user',
log: "/log", log: '/log',
midjourney: "/midjourney", midjourney: '/midjourney',
setting: "/setting", setting: '/setting',
about: "/about", about: '/about',
chat: "/chat", chat: '/chat',
detail: "/detail", detail: '/detail'
}; };
return ( return (
<Link <Link
style={{textDecoration: "none"}} style={{ textDecoration: 'none' }}
to={routerMap[props.itemKey]} to={routerMap[props.itemKey]}
> >
{itemElement} {itemElement}
</Link> </Link>
); );
}} }}
items={headerButtons} items={headerButtons}
onSelect={key => { onSelect={key => {
setSelectedKeys([key.itemKey]); setSelectedKeys([key.itemKey]);
}} }}
header={{ header={{
logo: <img src={logo} alt='logo' style={{marginRight: '0.75em'}}/>, logo: <img src={logo} alt="logo" style={{ marginRight: '0.75em' }} />,
text: systemName, text: systemName
}} }}
// footer={{ // footer={{
// text: '© 2021 NekoAPI', // text: '© 2021 NekoAPI',
// }} // }}
> >
<Nav.Footer collapseButton={true}> <Nav.Footer collapseButton={true}>
</Nav.Footer> </Nav.Footer>
</Nav> </Nav>
</div> </div>
</Layout> </Layout>
</> </>
); );
}; };
export default SiderBar; export default SiderBar;

File diff suppressed because it is too large Load Diff

File diff suppressed because it is too large Load Diff

View File

@ -1,337 +1,338 @@
import React, {useEffect, useState} from 'react'; import React, { useEffect, useState } from 'react';
import {API, isAdmin, showError, showSuccess} from '../helpers'; import { API, showError, showSuccess } from '../helpers';
import {Button, Modal, Popconfirm, Popover, Table, Tag, Form, Tooltip, Space} from "@douyinfe/semi-ui"; import { Button, Form, Popconfirm, Space, Table, Tag, Tooltip } from '@douyinfe/semi-ui';
import {ITEMS_PER_PAGE} from '../constants'; import { ITEMS_PER_PAGE } from '../constants';
import {renderGroup, renderNumber, renderQuota, renderText, stringToColor} from '../helpers/render'; import { renderGroup, renderNumber, renderQuota } from '../helpers/render';
import AddUser from "../pages/User/AddUser"; import AddUser from '../pages/User/AddUser';
import EditUser from "../pages/User/EditUser"; import EditUser from '../pages/User/EditUser';
function renderRole(role) { function renderRole(role) {
switch (role) { switch (role) {
case 1: case 1:
return <Tag size='large'>普通用户</Tag>; return <Tag size="large">普通用户</Tag>;
case 10: case 10:
return <Tag color='yellow' size='large'>管理员</Tag>; return <Tag color="yellow" size="large">管理员</Tag>;
case 100: case 100:
return <Tag color='orange' size='large'>超级管理员</Tag>; return <Tag color="orange" size="large">超级管理员</Tag>;
default: default:
return <Tag color='red' size='large'>未知身份</Tag>; return <Tag color="red" size="large">未知身份</Tag>;
} }
} }
const UsersTable = () => { const UsersTable = () => {
const columns = [{ const columns = [{
title: 'ID', dataIndex: 'id', title: 'ID', dataIndex: 'id'
}, { }, {
title: '用户名', dataIndex: 'username', title: '用户名', dataIndex: 'username'
}, { }, {
title: '分组', dataIndex: 'group', render: (text, record, index) => { title: '分组', dataIndex: 'group', render: (text, record, index) => {
return (<div> return (<div>
{renderGroup(text)} {renderGroup(text)}
</div>); </div>);
}, }
}, { }, {
title: '统计信息', dataIndex: 'info', render: (text, record, index) => { title: '统计信息', dataIndex: 'info', render: (text, record, index) => {
return (<div> return (<div>
<Space spacing={1}> <Space spacing={1}>
<Tooltip content={'剩余额度'}> <Tooltip content={'剩余额度'}>
<Tag color='white' size='large'>{renderQuota(record.quota)}</Tag> <Tag color="white" size="large">{renderQuota(record.quota)}</Tag>
</Tooltip> </Tooltip>
<Tooltip content={'已用额度'}> <Tooltip content={'已用额度'}>
<Tag color='white' size='large'>{renderQuota(record.used_quota)}</Tag> <Tag color="white" size="large">{renderQuota(record.used_quota)}</Tag>
</Tooltip> </Tooltip>
<Tooltip content={'调用次数'}> <Tooltip content={'调用次数'}>
<Tag color='white' size='large'>{renderNumber(record.request_count)}</Tag> <Tag color="white" size="large">{renderNumber(record.request_count)}</Tag>
</Tooltip> </Tooltip>
</Space> </Space>
</div>); </div>);
} }
}, { }, {
title: '邀请信息', dataIndex: 'invite', render: (text, record, index) => { title: '邀请信息', dataIndex: 'invite', render: (text, record, index) => {
return (<div> return (<div>
<Space spacing={1}> <Space spacing={1}>
<Tooltip content={'邀请人数'}> <Tooltip content={'邀请人数'}>
<Tag color='white' size='large'>{renderNumber(record.aff_count)}</Tag> <Tag color="white" size="large">{renderNumber(record.aff_count)}</Tag>
</Tooltip> </Tooltip>
<Tooltip content={'邀请总收益'}> <Tooltip content={'邀请总收益'}>
<Tag color='white' size='large'>{renderQuota(record.aff_history_quota)}</Tag> <Tag color="white" size="large">{renderQuota(record.aff_history_quota)}</Tag>
</Tooltip> </Tooltip>
<Tooltip content={'邀请人ID'}> <Tooltip content={'邀请人ID'}>
{record.inviter_id === 0 ? <Tag color='white' size='large'></Tag> : {record.inviter_id === 0 ? <Tag color="white" size="large"></Tag> :
<Tag color='white' size='large'>{record.inviter_id}</Tag>} <Tag color="white" size="large">{record.inviter_id}</Tag>}
</Tooltip> </Tooltip>
</Space> </Space>
</div>); </div>);
} }
}, { }, {
title: '角色', dataIndex: 'role', render: (text, record, index) => { title: '角色', dataIndex: 'role', render: (text, record, index) => {
return (<div> return (<div>
{renderRole(text)} {renderRole(text)}
</div>); </div>);
}, }
}, { }, {
title: '状态', dataIndex: 'status', render: (text, record, index) => { title: '状态', dataIndex: 'status', render: (text, record, index) => {
return (<div> return (<div>
{record.DeletedAt !== null? <Tag color='red'>已注销</Tag> : renderStatus(text)} {record.DeletedAt !== null ? <Tag color="red">已注销</Tag> : renderStatus(text)}
</div>); </div>);
}, }
}, { }, {
title: '', dataIndex: 'operate', render: (text, record, index) => (<div> title: '', dataIndex: 'operate', render: (text, record, index) => (<div>
{ {
record.DeletedAt !== null ? <></>: record.DeletedAt !== null ? <></> :
<> <>
<Popconfirm
title="确定?"
okType={'warning'}
onConfirm={() => {
manageUser(record.username, 'promote', record)
}}
>
<Button theme='light' type='warning' style={{marginRight: 1}}>提升</Button>
</Popconfirm>
<Popconfirm
title="确定?"
okType={'warning'}
onConfirm={() => {
manageUser(record.username, 'demote', record)
}}
>
<Button theme='light' type='secondary' style={{marginRight: 1}}>降级</Button>
</Popconfirm>
{record.status === 1 ?
<Button theme='light' type='warning' style={{marginRight: 1}} onClick={async () => {
manageUser(record.username, 'disable', record)
}}>禁用</Button> :
<Button theme='light' type='secondary' style={{marginRight: 1}} onClick={async () => {
manageUser(record.username, 'enable', record);
}} disabled={record.status === 3}>启用</Button>}
<Button theme='light' type='tertiary' style={{marginRight: 1}} onClick={() => {
setEditingUser(record);
setShowEditUser(true);
}}>编辑</Button>
</>
}
<Popconfirm <Popconfirm
title="确定是否要删除此用户?" title="确定?"
content="硬删除,此修改将不可逆" okType={'warning'}
okType={'danger'} onConfirm={() => {
position={'left'} manageUser(record.username, 'promote', record);
onConfirm={() => { }}
manageUser(record.username, 'delete', record).then(() => {
removeRecord(record.id);
})
}}
> >
<Button theme='light' type='danger' style={{marginRight: 1}}>删除</Button> <Button theme="light" type="warning" style={{ marginRight: 1 }}>提升</Button>
</Popconfirm> </Popconfirm>
</div>), <Popconfirm
},]; title="确定?"
okType={'warning'}
onConfirm={() => {
manageUser(record.username, 'demote', record);
}}
>
<Button theme="light" type="secondary" style={{ marginRight: 1 }}>降级</Button>
</Popconfirm>
{record.status === 1 ?
<Button theme="light" type="warning" style={{ marginRight: 1 }} onClick={async () => {
manageUser(record.username, 'disable', record);
}}>禁用</Button> :
<Button theme="light" type="secondary" style={{ marginRight: 1 }} onClick={async () => {
manageUser(record.username, 'enable', record);
}} disabled={record.status === 3}>启用</Button>}
<Button theme="light" type="tertiary" style={{ marginRight: 1 }} onClick={() => {
setEditingUser(record);
setShowEditUser(true);
}}>编辑</Button>
</>
const [users, setUsers] = useState([]); }
const [loading, setLoading] = useState(true); <Popconfirm
const [activePage, setActivePage] = useState(1); title="确定是否要删除此用户?"
const [searchKeyword, setSearchKeyword] = useState(''); content="硬删除,此修改将不可逆"
const [searching, setSearching] = useState(false); okType={'danger'}
const [userCount, setUserCount] = useState(ITEMS_PER_PAGE); position={'left'}
const [showAddUser, setShowAddUser] = useState(false); onConfirm={() => {
const [showEditUser, setShowEditUser] = useState(false); manageUser(record.username, 'delete', record).then(() => {
const [editingUser, setEditingUser] = useState({ removeRecord(record.id);
id: undefined, });
}}
>
<Button theme="light" type="danger" style={{ marginRight: 1 }}>删除</Button>
</Popconfirm>
</div>)
}];
const [users, setUsers] = useState([]);
const [loading, setLoading] = useState(true);
const [activePage, setActivePage] = useState(1);
const [searchKeyword, setSearchKeyword] = useState('');
const [searching, setSearching] = useState(false);
const [userCount, setUserCount] = useState(ITEMS_PER_PAGE);
const [showAddUser, setShowAddUser] = useState(false);
const [showEditUser, setShowEditUser] = useState(false);
const [editingUser, setEditingUser] = useState({
id: undefined
});
const setCount = (data) => {
if (data.length >= (activePage) * ITEMS_PER_PAGE) {
setUserCount(data.length + 1);
} else {
setUserCount(data.length);
}
};
const removeRecord = key => {
console.log(key);
let newDataSource = [...users];
if (key != null) {
let idx = newDataSource.findIndex(data => data.id === key);
if (idx > -1) {
newDataSource.splice(idx, 1);
setUsers(newDataSource);
}
}
};
const loadUsers = async (startIdx) => {
const res = await API.get(`/api/user/?p=${startIdx}`);
const { success, message, data } = res.data;
if (success) {
if (startIdx === 0) {
setUsers(data);
setCount(data);
} else {
let newUsers = users;
newUsers.push(...data);
setUsers(newUsers);
setCount(newUsers);
}
} else {
showError(message);
}
setLoading(false);
};
const onPaginationChange = (e, { activePage }) => {
(async () => {
if (activePage === Math.ceil(users.length / ITEMS_PER_PAGE) + 1) {
// In this case we have to load more data and then append them.
await loadUsers(activePage - 1);
}
setActivePage(activePage);
})();
};
useEffect(() => {
loadUsers(0)
.then()
.catch((reason) => {
showError(reason);
});
}, []);
const manageUser = async (username, action, record) => {
const res = await API.post('/api/user/manage', {
username, action
}); });
const { success, message } = res.data;
if (success) {
showSuccess('操作成功完成!');
let user = res.data.data;
let newUsers = [...users];
if (action === 'delete') {
const setCount = (data) => { } else {
if (data.length >= (activePage) * ITEMS_PER_PAGE) { record.status = user.status;
setUserCount(data.length + 1); record.role = user.role;
} else { }
setUserCount(data.length); setUsers(newUsers);
} } else {
showError(message);
} }
};
const removeRecord = key => { const renderStatus = (status) => {
console.log(key); switch (status) {
let newDataSource = [...users]; case 1:
if (key != null) { return <Tag size="large">已激活</Tag>;
let idx = newDataSource.findIndex(data => data.id === key); case 2:
return (<Tag size="large" color="red">
if (idx > -1) { 已封禁
newDataSource.splice(idx, 1); </Tag>);
setUsers(newDataSource); default:
} return (<Tag size="large" color="grey">
} 未知状态
}; </Tag>);
const loadUsers = async (startIdx) => {
const res = await API.get(`/api/user/?p=${startIdx}`);
const {success, message, data} = res.data;
if (success) {
if (startIdx === 0) {
setUsers(data);
setCount(data);
} else {
let newUsers = users;
newUsers.push(...data);
setUsers(newUsers);
setCount(newUsers);
}
} else {
showError(message);
}
setLoading(false);
};
const onPaginationChange = (e, {activePage}) => {
(async () => {
if (activePage === Math.ceil(users.length / ITEMS_PER_PAGE) + 1) {
// In this case we have to load more data and then append them.
await loadUsers(activePage - 1);
}
setActivePage(activePage);
})();
};
useEffect(() => {
loadUsers(0)
.then()
.catch((reason) => {
showError(reason);
});
}, []);
const manageUser = async (username, action, record) => {
const res = await API.post('/api/user/manage', {
username, action
});
const {success, message} = res.data;
if (success) {
showSuccess('操作成功完成!');
let user = res.data.data;
let newUsers = [...users];
if (action === 'delete') {
} else {
record.status = user.status;
record.role = user.role;
}
setUsers(newUsers);
} else {
showError(message);
}
};
const renderStatus = (status) => {
switch (status) {
case 1:
return <Tag size='large'>已激活</Tag>;
case 2:
return (<Tag size='large' color='red'>
已封禁
</Tag>);
default:
return (<Tag size='large' color='grey'>
未知状态
</Tag>);
}
};
const searchUsers = async () => {
if (searchKeyword === '') {
// if keyword is blank, load files instead.
await loadUsers(0);
setActivePage(1);
return;
}
setSearching(true);
const res = await API.get(`/api/user/search?keyword=${searchKeyword}`);
const {success, message, data} = res.data;
if (success) {
setUsers(data);
setActivePage(1);
} else {
showError(message);
}
setSearching(false);
};
const handleKeywordChange = async (value) => {
setSearchKeyword(value.trim());
};
const sortUser = (key) => {
if (users.length === 0) return;
setLoading(true);
let sortedUsers = [...users];
sortedUsers.sort((a, b) => {
return ('' + a[key]).localeCompare(b[key]);
});
if (sortedUsers[0].id === users[0].id) {
sortedUsers.reverse();
}
setUsers(sortedUsers);
setLoading(false);
};
const handlePageChange = page => {
setActivePage(page);
if (page === Math.ceil(users.length / ITEMS_PER_PAGE) + 1) {
// In this case we have to load more data and then append them.
loadUsers(page - 1).then(r => {
});
}
};
const pageData = users.slice((activePage - 1) * ITEMS_PER_PAGE, activePage * ITEMS_PER_PAGE);
const closeAddUser = () => {
setShowAddUser(false);
} }
};
const closeEditUser = () => { const searchUsers = async () => {
setShowEditUser(false); if (searchKeyword === '') {
setEditingUser({ // if keyword is blank, load files instead.
id: undefined, await loadUsers(0);
}); setActivePage(1);
return;
} }
setSearching(true);
const res = await API.get(`/api/user/search?keyword=${searchKeyword}`);
const { success, message, data } = res.data;
if (success) {
setUsers(data);
setActivePage(1);
} else {
showError(message);
}
setSearching(false);
};
const refresh = async () => { const handleKeywordChange = async (value) => {
if (searchKeyword === '') { setSearchKeyword(value.trim());
await loadUsers(activePage - 1); };
} else {
await searchUsers(); const sortUser = (key) => {
if (users.length === 0) return;
setLoading(true);
let sortedUsers = [...users];
sortedUsers.sort((a, b) => {
return ('' + a[key]).localeCompare(b[key]);
});
if (sortedUsers[0].id === users[0].id) {
sortedUsers.reverse();
}
setUsers(sortedUsers);
setLoading(false);
};
const handlePageChange = page => {
setActivePage(page);
if (page === Math.ceil(users.length / ITEMS_PER_PAGE) + 1) {
// In this case we have to load more data and then append them.
loadUsers(page - 1).then(r => {
});
}
};
const pageData = users.slice((activePage - 1) * ITEMS_PER_PAGE, activePage * ITEMS_PER_PAGE);
const closeAddUser = () => {
setShowAddUser(false);
};
const closeEditUser = () => {
setShowEditUser(false);
setEditingUser({
id: undefined
});
};
const refresh = async () => {
if (searchKeyword === '') {
await loadUsers(activePage - 1);
} else {
await searchUsers();
}
};
return (
<>
<AddUser refresh={refresh} visible={showAddUser} handleClose={closeAddUser}></AddUser>
<EditUser refresh={refresh} visible={showEditUser} handleClose={closeEditUser}
editingUser={editingUser}></EditUser>
<Form onSubmit={searchUsers}>
<Form.Input
label="搜索关键字"
icon="search"
field="keyword"
iconPosition="left"
placeholder="搜索用户的 ID用户名显示名称以及邮箱地址 ..."
value={searchKeyword}
loading={searching}
onChange={value => handleKeywordChange(value)}
/>
</Form>
<Table columns={columns} dataSource={pageData} pagination={{
currentPage: activePage,
pageSize: ITEMS_PER_PAGE,
total: userCount,
pageSizeOpts: [10, 20, 50, 100],
onPageChange: handlePageChange
}} loading={loading} />
<Button theme="light" type="primary" style={{ marginRight: 8 }} onClick={
() => {
setShowAddUser(true);
} }
}; }>添加用户</Button>
</>
return ( );
<>
<AddUser refresh={refresh} visible={showAddUser} handleClose={closeAddUser}></AddUser>
<EditUser refresh={refresh} visible={showEditUser} handleClose={closeEditUser} editingUser={editingUser}></EditUser>
<Form onSubmit={searchUsers}>
<Form.Input
label='搜索关键字'
icon='search'
field='keyword'
iconPosition='left'
placeholder='搜索用户的 ID用户名显示名称以及邮箱地址 ...'
value={searchKeyword}
loading={searching}
onChange={value => handleKeywordChange(value)}
/>
</Form>
<Table columns={columns} dataSource={pageData} pagination={{
currentPage: activePage,
pageSize: ITEMS_PER_PAGE,
total: userCount,
pageSizeOpts: [10, 20, 50, 100],
onPageChange: handlePageChange,
}} loading={loading}/>
<Button theme='light' type='primary' style={{marginRight: 8}} onClick={
() => {
setShowAddUser(true);
}
}>添加用户</Button>
</>
);
}; };
export default UsersTable; export default UsersTable;

View File

@ -3,14 +3,14 @@ import { Icon } from '@douyinfe/semi-ui';
const WeChatIcon = () => { const WeChatIcon = () => {
function CustomIcon() { function CustomIcon() {
return <svg t='1709714447384' className='icon' viewBox='0 0 1024 1024' version='1.1' return <svg t="1709714447384" className="icon" viewBox="0 0 1024 1024" version="1.1"
xmlns='http://www.w3.org/2000/svg' p-id='5091' width='16' height='16'> xmlns="http://www.w3.org/2000/svg" p-id="5091" width="16" height="16">
<path <path
d='M690.1 377.4c5.9 0 11.8 0.2 17.6 0.5-24.4-128.7-158.3-227.1-319.9-227.1C209 150.8 64 271.4 64 420.2c0 81.1 43.6 154.2 111.9 203.6 5.5 3.9 9.1 10.3 9.1 17.6 0 2.4-0.5 4.6-1.1 6.9-5.5 20.3-14.2 52.8-14.6 54.3-0.7 2.6-1.7 5.2-1.7 7.9 0 5.9 4.8 10.8 10.8 10.8 2.3 0 4.2-0.9 6.2-2l70.9-40.9c5.3-3.1 11-5 17.2-5 3.2 0 6.4 0.5 9.5 1.4 33.1 9.5 68.8 14.8 105.7 14.8 6 0 11.9-0.1 17.8-0.4-7.1-21-10.9-43.1-10.9-66 0-135.8 132.2-245.8 295.3-245.8z m-194.3-86.5c23.8 0 43.2 19.3 43.2 43.1s-19.3 43.1-43.2 43.1c-23.8 0-43.2-19.3-43.2-43.1s19.4-43.1 43.2-43.1z m-215.9 86.2c-23.8 0-43.2-19.3-43.2-43.1s19.3-43.1 43.2-43.1 43.2 19.3 43.2 43.1-19.4 43.1-43.2 43.1z' d="M690.1 377.4c5.9 0 11.8 0.2 17.6 0.5-24.4-128.7-158.3-227.1-319.9-227.1C209 150.8 64 271.4 64 420.2c0 81.1 43.6 154.2 111.9 203.6 5.5 3.9 9.1 10.3 9.1 17.6 0 2.4-0.5 4.6-1.1 6.9-5.5 20.3-14.2 52.8-14.6 54.3-0.7 2.6-1.7 5.2-1.7 7.9 0 5.9 4.8 10.8 10.8 10.8 2.3 0 4.2-0.9 6.2-2l70.9-40.9c5.3-3.1 11-5 17.2-5 3.2 0 6.4 0.5 9.5 1.4 33.1 9.5 68.8 14.8 105.7 14.8 6 0 11.9-0.1 17.8-0.4-7.1-21-10.9-43.1-10.9-66 0-135.8 132.2-245.8 295.3-245.8z m-194.3-86.5c23.8 0 43.2 19.3 43.2 43.1s-19.3 43.1-43.2 43.1c-23.8 0-43.2-19.3-43.2-43.1s19.4-43.1 43.2-43.1z m-215.9 86.2c-23.8 0-43.2-19.3-43.2-43.1s19.3-43.1 43.2-43.1 43.2 19.3 43.2 43.1-19.4 43.1-43.2 43.1z"
p-id='5092'></path> p-id="5092"></path>
<path <path
d='M866.7 792.7c56.9-41.2 93.2-102 93.2-169.7 0-124-120.8-224.5-269.9-224.5-149 0-269.9 100.5-269.9 224.5S540.9 847.5 690 847.5c30.8 0 60.6-4.4 88.1-12.3 2.6-0.8 5.2-1.2 7.9-1.2 5.2 0 9.9 1.6 14.3 4.1l59.1 34c1.7 1 3.3 1.7 5.2 1.7 2.4 0 4.7-0.9 6.4-2.6 1.7-1.7 2.6-4 2.6-6.4 0-2.2-0.9-4.4-1.4-6.6-0.3-1.2-7.6-28.3-12.2-45.3-0.5-1.9-0.9-3.8-0.9-5.7 0.1-5.9 3.1-11.2 7.6-14.5zM600.2 587.2c-19.9 0-36-16.1-36-35.9 0-19.8 16.1-35.9 36-35.9s36 16.1 36 35.9c0 19.8-16.2 35.9-36 35.9z m179.9 0c-19.9 0-36-16.1-36-35.9 0-19.8 16.1-35.9 36-35.9s36 16.1 36 35.9c-0.1 19.8-16.2 35.9-36 35.9z' d="M866.7 792.7c56.9-41.2 93.2-102 93.2-169.7 0-124-120.8-224.5-269.9-224.5-149 0-269.9 100.5-269.9 224.5S540.9 847.5 690 847.5c30.8 0 60.6-4.4 88.1-12.3 2.6-0.8 5.2-1.2 7.9-1.2 5.2 0 9.9 1.6 14.3 4.1l59.1 34c1.7 1 3.3 1.7 5.2 1.7 2.4 0 4.7-0.9 6.4-2.6 1.7-1.7 2.6-4 2.6-6.4 0-2.2-0.9-4.4-1.4-6.6-0.3-1.2-7.6-28.3-12.2-45.3-0.5-1.9-0.9-3.8-0.9-5.7 0.1-5.9 3.1-11.2 7.6-14.5zM600.2 587.2c-19.9 0-36-16.1-36-35.9 0-19.8 16.1-35.9 36-35.9s36 16.1 36 35.9c0 19.8-16.2 35.9-36 35.9z m179.9 0c-19.9 0-36-16.1-36-35.9 0-19.8 16.1-35.9 36-35.9s36 16.1 36 35.9c-0.1 19.8-16.2 35.9-36 35.9z"
p-id='5093'></path> p-id="5093"></path>
</svg>; </svg>;
} }

View File

@ -1,17 +1,17 @@
import React, { useEffect, useState } from 'react'; import React, { useEffect, useState } from 'react';
import { useParams, useNavigate } from 'react-router-dom'; import { useNavigate, useParams } from 'react-router-dom';
import {API, downloadTextAsFile, isMobile, showError, showSuccess} from '../../helpers'; import { API, downloadTextAsFile, isMobile, showError, showSuccess } from '../../helpers';
import { renderQuota, renderQuotaWithPrompt } from '../../helpers/render'; import { renderQuotaWithPrompt } from '../../helpers/render';
import {SideSheet, Space, Spin, Button, Input, Typography, AutoComplete, Modal} from "@douyinfe/semi-ui"; import { AutoComplete, Button, Input, Modal, SideSheet, Space, Spin, Typography } from '@douyinfe/semi-ui';
import Title from "@douyinfe/semi-ui/lib/es/typography/title"; import Title from '@douyinfe/semi-ui/lib/es/typography/title';
import {Divider} from "semantic-ui-react"; import { Divider } from 'semantic-ui-react';
const EditRedemption = (props) => { const EditRedemption = (props) => {
const isEdit = props.editingRedemption.id !== undefined; const isEdit = props.editingRedemption.id !== undefined;
const [loading, setLoading] = useState(isEdit); const [loading, setLoading] = useState(isEdit);
const params = useParams(); const params = useParams();
const navigate = useNavigate() const navigate = useNavigate();
const originInputs = { const originInputs = {
name: '', name: '',
quota: 100000, quota: 100000,
@ -22,8 +22,8 @@ const EditRedemption = (props) => {
const handleCancel = () => { const handleCancel = () => {
props.handleClose(); props.handleClose();
} };
const handleInputChange = (name, value) => { const handleInputChange = (name, value) => {
setInputs((inputs) => ({ ...inputs, [name]: value })); setInputs((inputs) => ({ ...inputs, [name]: value }));
}; };
@ -43,9 +43,9 @@ const EditRedemption = (props) => {
useEffect(() => { useEffect(() => {
if (isEdit) { if (isEdit) {
loadRedemption().then( loadRedemption().then(
() => { () => {
// console.log(inputs); // console.log(inputs);
} }
); );
} else { } else {
setInputs(originInputs); setInputs(originInputs);
@ -82,21 +82,21 @@ const EditRedemption = (props) => {
showError(message); showError(message);
} }
if (!isEdit && data) { if (!isEdit && data) {
let text = ""; let text = '';
for (let i = 0; i < data.length; i++) { for (let i = 0; i < data.length; i++) {
text += data[i] + "\n"; text += data[i] + '\n';
} }
// downloadTextAsFile(text, `${inputs.name}.txt`); // downloadTextAsFile(text, `${inputs.name}.txt`);
Modal.confirm({ Modal.confirm({
title: '兑换码创建成功', title: '兑换码创建成功',
content: ( content: (
<div> <div>
<p>兑换码创建成功是否下载兑换码</p> <p>兑换码创建成功是否下载兑换码</p>
<p>兑换码将以文本文件的形式下载文件名为兑换码的名称</p> <p>兑换码将以文本文件的形式下载文件名为兑换码的名称</p>
</div> </div>
), ),
onOk: () => { onOk: () => {
downloadTextAsFile(text, `${inputs.name}.txt`); downloadTextAsFile(text, `${inputs.name}.txt`);
} }
}); });
} }
@ -106,71 +106,71 @@ const EditRedemption = (props) => {
return ( return (
<> <>
<SideSheet <SideSheet
placement={isEdit ? 'right' : 'left'} placement={isEdit ? 'right' : 'left'}
title={<Title level={3}>{isEdit ? '更新兑换码信息' : '创建新的兑换码'}</Title>} title={<Title level={3}>{isEdit ? '更新兑换码信息' : '创建新的兑换码'}</Title>}
headerStyle={{borderBottom: '1px solid var(--semi-color-border)'}} headerStyle={{ borderBottom: '1px solid var(--semi-color-border)' }}
bodyStyle={{borderBottom: '1px solid var(--semi-color-border)'}} bodyStyle={{ borderBottom: '1px solid var(--semi-color-border)' }}
visible={props.visiable} visible={props.visiable}
footer={ footer={
<div style={{display: 'flex', justifyContent: 'flex-end'}}> <div style={{ display: 'flex', justifyContent: 'flex-end' }}>
<Space> <Space>
<Button theme='solid' size={'large'} onClick={submit}>提交</Button> <Button theme="solid" size={'large'} onClick={submit}>提交</Button>
<Button theme='solid' size={'large'} type={'tertiary'} onClick={handleCancel}>取消</Button> <Button theme="solid" size={'large'} type={'tertiary'} onClick={handleCancel}>取消</Button>
</Space> </Space>
</div> </div>
} }
closeIcon={null} closeIcon={null}
onCancel={() => handleCancel()} onCancel={() => handleCancel()}
width={isMobile() ? '100%' : 600} width={isMobile() ? '100%' : 600}
> >
<Spin spinning={loading}> <Spin spinning={loading}>
<Input <Input
style={{ marginTop: 20 }} style={{ marginTop: 20 }}
label='名称' label="名称"
name='name' name="name"
placeholder={'请输入名称'} placeholder={'请输入名称'}
onChange={value => handleInputChange('name', value)} onChange={value => handleInputChange('name', value)}
value={name} value={name}
autoComplete='new-password' autoComplete="new-password"
required={!isEdit} required={!isEdit}
/> />
<Divider/> <Divider />
<div style={{ marginTop: 20 }}> <div style={{ marginTop: 20 }}>
<Typography.Text>{`额度${renderQuotaWithPrompt(quota)}`}</Typography.Text> <Typography.Text>{`额度${renderQuotaWithPrompt(quota)}`}</Typography.Text>
</div> </div>
<AutoComplete <AutoComplete
style={{ marginTop: 8 }} style={{ marginTop: 8 }}
name='quota' name="quota"
placeholder={'请输入额度'} placeholder={'请输入额度'}
onChange={(value) => handleInputChange('quota', value)} onChange={(value) => handleInputChange('quota', value)}
value={quota} value={quota}
autoComplete='new-password' autoComplete="new-password"
type='number' type="number"
position={'bottom'} position={'bottom'}
data={[ data={[
{value: 500000, label: '1$'}, { value: 500000, label: '1$' },
{value: 5000000, label: '10$'}, { value: 5000000, label: '10$' },
{value: 25000000, label: '50$'}, { value: 25000000, label: '50$' },
{value: 50000000, label: '100$'}, { value: 50000000, label: '100$' },
{value: 250000000, label: '500$'}, { value: 250000000, label: '500$' },
{value: 500000000, label: '1000$'}, { value: 500000000, label: '1000$' }
]} ]}
/> />
{ {
!isEdit && <> !isEdit && <>
<Divider/> <Divider />
<Typography.Text>生成数量</Typography.Text> <Typography.Text>生成数量</Typography.Text>
<Input <Input
style={{ marginTop: 8 }} style={{ marginTop: 8 }}
label='生成数量' label="生成数量"
name='count' name="count"
placeholder={'请输入生成数量'} placeholder={'请输入生成数量'}
onChange={value => handleInputChange('count', value)} onChange={value => handleInputChange('count', value)}
value={count} value={count}
autoComplete='new-password' autoComplete="new-password"
type='number' type="number"
/> />
</> </>
} }
</Spin> </Spin>
</SideSheet> </SideSheet>

View File

@ -1,352 +1,351 @@
import React, {useEffect, useRef, useState} from 'react'; import React, { useEffect, useState } from 'react';
import {useParams, useNavigate} from 'react-router-dom'; import { useNavigate } from 'react-router-dom';
import {API, isMobile, showError, showSuccess, timestamp2string} from '../../helpers'; import { API, isMobile, showError, showSuccess, timestamp2string } from '../../helpers';
import {renderQuota, renderQuotaWithPrompt} from '../../helpers/render'; import { renderQuotaWithPrompt } from '../../helpers/render';
import { import {
Layout, AutoComplete,
SideSheet, Banner,
Button, Button,
Checkbox,
DatePicker,
Input,
Select,
SideSheet,
Space, Space,
Spin, Spin,
Banner, Typography
Input, } from '@douyinfe/semi-ui';
DatePicker, import Title from '@douyinfe/semi-ui/lib/es/typography/title';
AutoComplete, import { Divider } from 'semantic-ui-react';
Typography,
Checkbox, Select
} from "@douyinfe/semi-ui";
import Title from "@douyinfe/semi-ui/lib/es/typography/title";
import {Divider} from "semantic-ui-react";
const EditToken = (props) => { const EditToken = (props) => {
const [isEdit, setIsEdit] = useState(false); const [isEdit, setIsEdit] = useState(false);
const [loading, setLoading] = useState(isEdit); const [loading, setLoading] = useState(isEdit);
const originInputs = { const originInputs = {
name: '', name: '',
remain_quota: isEdit ? 0 : 500000, remain_quota: isEdit ? 0 : 500000,
expired_time: -1, expired_time: -1,
unlimited_quota: false, unlimited_quota: false,
model_limits_enabled: false, model_limits_enabled: false,
model_limits: [], model_limits: []
}; };
const [inputs, setInputs] = useState(originInputs); const [inputs, setInputs] = useState(originInputs);
const {name, remain_quota, expired_time, unlimited_quota, model_limits_enabled, model_limits} = inputs; const { name, remain_quota, expired_time, unlimited_quota, model_limits_enabled, model_limits } = inputs;
// const [visible, setVisible] = useState(false); // const [visible, setVisible] = useState(false);
const [models, setModels] = useState({}); const [models, setModels] = useState({});
const navigate = useNavigate(); const navigate = useNavigate();
const handleInputChange = (name, value) => { const handleInputChange = (name, value) => {
setInputs((inputs) => ({...inputs, [name]: value})); setInputs((inputs) => ({ ...inputs, [name]: value }));
}; };
const handleCancel = () => { const handleCancel = () => {
props.handleClose();
};
const setExpiredTime = (month, day, hour, minute) => {
let now = new Date();
let timestamp = now.getTime() / 1000;
let seconds = month * 30 * 24 * 60 * 60;
seconds += day * 24 * 60 * 60;
seconds += hour * 60 * 60;
seconds += minute * 60;
if (seconds !== 0) {
timestamp += seconds;
setInputs({ ...inputs, expired_time: timestamp2string(timestamp) });
} else {
setInputs({ ...inputs, expired_time: -1 });
}
};
const setUnlimitedQuota = () => {
setInputs({ ...inputs, unlimited_quota: !unlimited_quota });
};
const loadModels = async () => {
let res = await API.get(`/api/user/models`);
const { success, message, data } = res.data;
if (success) {
let localModelOptions = data.map((model) => ({
label: model,
value: model
}));
setModels(localModelOptions);
} else {
showError(message);
}
};
const loadToken = async () => {
setLoading(true);
let res = await API.get(`/api/token/${props.editingToken.id}`);
const { success, message, data } = res.data;
if (success) {
if (data.expired_time !== -1) {
data.expired_time = timestamp2string(data.expired_time);
}
if (data.model_limits !== '') {
data.model_limits = data.model_limits.split(',');
} else {
data.model_limits = [];
}
setInputs(data);
} else {
showError(message);
}
setLoading(false);
};
useEffect(() => {
setIsEdit(props.editingToken.id !== undefined);
}, [props.editingToken.id]);
useEffect(() => {
if (!isEdit) {
setInputs(originInputs);
} else {
loadToken().then(
() => {
// console.log(inputs);
}
);
}
loadModels();
}, [isEdit]);
// 新增 state 变量 tokenCount 来记录用户想要创建的令牌数量,默认为 1
const [tokenCount, setTokenCount] = useState(1);
// 新增处理 tokenCount 变化的函数
const handleTokenCountChange = (value) => {
// 确保用户输入的是正整数
const count = parseInt(value, 10);
if (!isNaN(count) && count > 0) {
setTokenCount(count);
}
};
// 生成一个随机的四位字母数字字符串
const generateRandomSuffix = () => {
const characters = 'ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789';
let result = '';
for (let i = 0; i < 6; i++) {
result += characters.charAt(Math.floor(Math.random() * characters.length));
}
return result;
};
const submit = async () => {
setLoading(true);
if (isEdit) {
// 编辑令牌的逻辑保持不变
let localInputs = { ...inputs };
localInputs.remain_quota = parseInt(localInputs.remain_quota);
if (localInputs.expired_time !== -1) {
let time = Date.parse(localInputs.expired_time);
if (isNaN(time)) {
showError('过期时间格式错误!');
setLoading(false);
return;
}
localInputs.expired_time = Math.ceil(time / 1000);
}
localInputs.model_limits = localInputs.model_limits.join(',');
let res = await API.put(`/api/token/`, { ...localInputs, id: parseInt(props.editingToken.id) });
const { success, message } = res.data;
if (success) {
showSuccess('令牌更新成功!');
props.refresh();
props.handleClose(); props.handleClose();
} } else {
const setExpiredTime = (month, day, hour, minute) => { showError(message);
let now = new Date(); }
let timestamp = now.getTime() / 1000; } else {
let seconds = month * 30 * 24 * 60 * 60; // 处理新增多个令牌的情况
seconds += day * 24 * 60 * 60; let successCount = 0; // 记录成功创建的令牌数量
seconds += hour * 60 * 60; for (let i = 0; i < tokenCount; i++) {
seconds += minute * 60; let localInputs = { ...inputs };
if (seconds !== 0) { if (i !== 0) {
timestamp += seconds; // 如果用户想要创建多个令牌,则给每个令牌一个序号后缀
setInputs({...inputs, expired_time: timestamp2string(timestamp)}); localInputs.name = `${inputs.name}-${generateRandomSuffix()}`;
} else {
setInputs({...inputs, expired_time: -1});
} }
}; localInputs.remain_quota = parseInt(localInputs.remain_quota);
const setUnlimitedQuota = () => { if (localInputs.expired_time !== -1) {
setInputs({...inputs, unlimited_quota: !unlimited_quota}); let time = Date.parse(localInputs.expired_time);
}; if (isNaN(time)) {
showError('过期时间格式错误!');
setLoading(false);
break;
}
localInputs.expired_time = Math.ceil(time / 1000);
}
localInputs.model_limits = localInputs.model_limits.join(',');
let res = await API.post(`/api/token/`, localInputs);
const { success, message } = res.data;
const loadModels = async () => {
let res = await API.get(`/api/user/models`);
const {success, message, data} = res.data;
if (success) { if (success) {
let localModelOptions = data.map((model) => ({ successCount++;
label: model,
value: model
}));
setModels(localModelOptions);
} else { } else {
showError(message); showError(message);
break; // 如果创建失败,终止循环
} }
}
if (successCount > 0) {
showSuccess(`${successCount}个令牌创建成功,请在列表页面点击复制获取令牌!`);
props.refresh();
props.handleClose();
}
} }
setLoading(false);
setInputs(originInputs); // 重置表单
setTokenCount(1); // 重置数量为默认值
};
const loadToken = async () => {
setLoading(true); return (
let res = await API.get(`/api/token/${props.editingToken.id}`); <>
const {success, message, data} = res.data; <SideSheet
if (success) { placement={isEdit ? 'right' : 'left'}
if (data.expired_time !== -1) { title={<Title level={3}>{isEdit ? '更新令牌信息' : '创建新的令牌'}</Title>}
data.expired_time = timestamp2string(data.expired_time); headerStyle={{ borderBottom: '1px solid var(--semi-color-border)' }}
} bodyStyle={{ borderBottom: '1px solid var(--semi-color-border)' }}
if (data.model_limits !== '') { visible={props.visiable}
data.model_limits = data.model_limits.split(','); footer={
} else { <div style={{ display: 'flex', justifyContent: 'flex-end' }}>
data.model_limits = []; <Space>
} <Button theme="solid" size={'large'} onClick={submit}>提交</Button>
setInputs(data); <Button theme="solid" size={'large'} type={'tertiary'} onClick={handleCancel}>取消</Button>
} else { </Space>
showError(message); </div>
} }
setLoading(false); closeIcon={null}
}; onCancel={() => handleCancel()}
useEffect(() => { width={isMobile() ? '100%' : 600}
setIsEdit(props.editingToken.id !== undefined); >
}, [props.editingToken.id]); <Spin spinning={loading}>
<Input
style={{ marginTop: 20 }}
label="名称"
name="name"
placeholder={'请输入名称'}
onChange={(value) => handleInputChange('name', value)}
value={name}
autoComplete="new-password"
required={!isEdit}
/>
<Divider />
<DatePicker
label="过期时间"
name="expired_time"
placeholder={'请选择过期时间'}
onChange={(value) => handleInputChange('expired_time', value)}
value={expired_time}
autoComplete="new-password"
type="dateTime"
/>
<div style={{ marginTop: 20 }}>
<Space>
<Button type={'tertiary'} onClick={() => {
setExpiredTime(0, 0, 0, 0);
}}>永不过期</Button>
<Button type={'tertiary'} onClick={() => {
setExpiredTime(0, 0, 1, 0);
}}>一小时</Button>
<Button type={'tertiary'} onClick={() => {
setExpiredTime(1, 0, 0, 0);
}}>一个月</Button>
<Button type={'tertiary'} onClick={() => {
setExpiredTime(0, 1, 0, 0);
}}>一天</Button>
</Space>
</div>
useEffect(() => { <Divider />
if (!isEdit) { <Banner type={'warning'}
setInputs(originInputs); description={'注意,令牌的额度仅用于限制令牌本身的最大额度使用量,实际的使用受到账户的剩余额度限制。'}></Banner>
} else { <div style={{ marginTop: 20 }}>
loadToken().then( <Typography.Text>{`额度${renderQuotaWithPrompt(remain_quota)}`}</Typography.Text>
() => { </div>
// console.log(inputs); <AutoComplete
} style={{ marginTop: 8 }}
); name="remain_quota"
} placeholder={'请输入额度'}
loadModels(); onChange={(value) => handleInputChange('remain_quota', value)}
}, [isEdit]); value={remain_quota}
autoComplete="new-password"
type="number"
// position={'top'}
data={[
{ value: 500000, label: '1$' },
{ value: 5000000, label: '10$' },
{ value: 25000000, label: '50$' },
{ value: 50000000, label: '100$' },
{ value: 250000000, label: '500$' },
{ value: 500000000, label: '1000$' }
]}
disabled={unlimited_quota}
/>
// 新增 state 变量 tokenCount 来记录用户想要创建的令牌数量,默认为 1 {!isEdit && (
const [tokenCount, setTokenCount] = useState(1); <>
<div style={{ marginTop: 20 }}>
<Typography.Text>新建数量</Typography.Text>
</div>
<AutoComplete
style={{ marginTop: 8 }}
label="数量"
placeholder={'请选择或输入创建令牌的数量'}
onChange={(value) => handleTokenCountChange(value)}
onSelect={(value) => handleTokenCountChange(value)}
value={tokenCount.toString()}
autoComplete="off"
type="number"
data={[
{ value: 10, label: '10个' },
{ value: 20, label: '20个' },
{ value: 30, label: '30个' },
{ value: 100, label: '100个' }
]}
disabled={unlimited_quota}
/>
</>
)}
// 新增处理 tokenCount 变化的函数 <div>
const handleTokenCountChange = (value) => { <Button style={{ marginTop: 8 }} type={'warning'} onClick={() => {
// 确保用户输入的是正整数 setUnlimitedQuota();
const count = parseInt(value, 10); }}>{unlimited_quota ? '取消无限额度' : '设为无限额度'}</Button>
if (!isNaN(count) && count > 0) { </div>
setTokenCount(count); <Divider />
} <div style={{ marginTop: 10, display: 'flex' }}>
}; <Space>
<Checkbox
name="model_limits_enabled"
checked={model_limits_enabled}
onChange={(e) => handleInputChange('model_limits_enabled', e.target.checked)}
>
</Checkbox>
<Typography.Text>启用模型限制非必要不建议启用</Typography.Text>
</Space>
</div>
// 生成一个随机的四位字母数字字符串 <Select
const generateRandomSuffix = () => { style={{ marginTop: 8 }}
const characters = 'ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789'; placeholder={'请选择该渠道所支持的模型'}
let result = ''; name="models"
for (let i = 0; i < 6; i++) { required
result += characters.charAt(Math.floor(Math.random() * characters.length)); multiple
} selection
return result; onChange={value => {
}; handleInputChange('model_limits', value);
}}
const submit = async () => { value={inputs.model_limits}
setLoading(true); autoComplete="new-password"
if (isEdit) { optionList={models}
// 编辑令牌的逻辑保持不变 disabled={!model_limits_enabled}
let localInputs = {...inputs}; />
localInputs.remain_quota = parseInt(localInputs.remain_quota); </Spin>
if (localInputs.expired_time !== -1) { </SideSheet>
let time = Date.parse(localInputs.expired_time); </>
if (isNaN(time)) { );
showError('过期时间格式错误!');
setLoading(false);
return;
}
localInputs.expired_time = Math.ceil(time / 1000);
}
localInputs.model_limits = localInputs.model_limits.join(',');
let res = await API.put(`/api/token/`, {...localInputs, id: parseInt(props.editingToken.id)});
const {success, message} = res.data;
if (success) {
showSuccess('令牌更新成功!');
props.refresh();
props.handleClose();
} else {
showError(message);
}
} else {
// 处理新增多个令牌的情况
let successCount = 0; // 记录成功创建的令牌数量
for (let i = 0; i < tokenCount; i++) {
let localInputs = {...inputs};
if (i !== 0) {
// 如果用户想要创建多个令牌,则给每个令牌一个序号后缀
localInputs.name = `${inputs.name}-${generateRandomSuffix()}`;
}
localInputs.remain_quota = parseInt(localInputs.remain_quota);
if (localInputs.expired_time !== -1) {
let time = Date.parse(localInputs.expired_time);
if (isNaN(time)) {
showError('过期时间格式错误!');
setLoading(false);
break;
}
localInputs.expired_time = Math.ceil(time / 1000);
}
localInputs.model_limits = localInputs.model_limits.join(',');
let res = await API.post(`/api/token/`, localInputs);
const {success, message} = res.data;
if (success) {
successCount++;
} else {
showError(message);
break; // 如果创建失败,终止循环
}
}
if (successCount > 0) {
showSuccess(`${successCount}个令牌创建成功,请在列表页面点击复制获取令牌!`);
props.refresh();
props.handleClose();
}
}
setLoading(false);
setInputs(originInputs); // 重置表单
setTokenCount(1); // 重置数量为默认值
};
return (
<>
<SideSheet
placement={isEdit ? 'right' : 'left'}
title={<Title level={3}>{isEdit ? '更新令牌信息' : '创建新的令牌'}</Title>}
headerStyle={{borderBottom: '1px solid var(--semi-color-border)'}}
bodyStyle={{borderBottom: '1px solid var(--semi-color-border)'}}
visible={props.visiable}
footer={
<div style={{display: 'flex', justifyContent: 'flex-end'}}>
<Space>
<Button theme='solid' size={'large'} onClick={submit}>提交</Button>
<Button theme='solid' size={'large'} type={'tertiary'} onClick={handleCancel}>取消</Button>
</Space>
</div>
}
closeIcon={null}
onCancel={() => handleCancel()}
width={isMobile() ? '100%' : 600}
>
<Spin spinning={loading}>
<Input
style={{marginTop: 20}}
label='名称'
name='name'
placeholder={'请输入名称'}
onChange={(value) => handleInputChange('name', value)}
value={name}
autoComplete='new-password'
required={!isEdit}
/>
<Divider/>
<DatePicker
label='过期时间'
name='expired_time'
placeholder={'请选择过期时间'}
onChange={(value) => handleInputChange('expired_time', value)}
value={expired_time}
autoComplete='new-password'
type='dateTime'
/>
<div style={{marginTop: 20}}>
<Space>
<Button type={'tertiary'} onClick={() => {
setExpiredTime(0, 0, 0, 0);
}}>永不过期</Button>
<Button type={'tertiary'} onClick={() => {
setExpiredTime(0, 0, 1, 0);
}}>一小时</Button>
<Button type={'tertiary'} onClick={() => {
setExpiredTime(1, 0, 0, 0);
}}>一个月</Button>
<Button type={'tertiary'} onClick={() => {
setExpiredTime(0, 1, 0, 0);
}}>一天</Button>
</Space>
</div>
<Divider/>
<Banner type={'warning'}
description={'注意,令牌的额度仅用于限制令牌本身的最大额度使用量,实际的使用受到账户的剩余额度限制。'}></Banner>
<div style={{marginTop: 20}}>
<Typography.Text>{`额度${renderQuotaWithPrompt(remain_quota)}`}</Typography.Text>
</div>
<AutoComplete
style={{marginTop: 8}}
name='remain_quota'
placeholder={'请输入额度'}
onChange={(value) => handleInputChange('remain_quota', value)}
value={remain_quota}
autoComplete='new-password'
type='number'
// position={'top'}
data={[
{value: 500000, label: '1$'},
{value: 5000000, label: '10$'},
{value: 25000000, label: '50$'},
{value: 50000000, label: '100$'},
{value: 250000000, label: '500$'},
{value: 500000000, label: '1000$'},
]}
disabled={unlimited_quota}
/>
{!isEdit && (
<>
<div style={{marginTop: 20}}>
<Typography.Text>新建数量</Typography.Text>
</div>
<AutoComplete
style={{ marginTop: 8 }}
label='数量'
placeholder={'请选择或输入创建令牌的数量'}
onChange={(value) => handleTokenCountChange(value)}
onSelect={(value) => handleTokenCountChange(value)}
value={tokenCount.toString()}
autoComplete='off'
type='number'
data={[
{ value: 10, label: '10个' },
{ value: 20, label: '20个' },
{ value: 30, label: '30个' },
{ value: 100, label: '100个' },
]}
disabled={unlimited_quota}
/>
</>
)}
<div>
<Button style={{marginTop: 8}} type={'warning'} onClick={() => {
setUnlimitedQuota();
}}>{unlimited_quota ? '取消无限额度' : '设为无限额度'}</Button>
</div>
<Divider/>
<div style={{marginTop: 10, display: 'flex'}}>
<Space>
<Checkbox
name='model_limits_enabled'
checked={model_limits_enabled}
onChange={(e) => handleInputChange('model_limits_enabled', e.target.checked)}
>
</Checkbox>
<Typography.Text>启用模型限制非必要不建议启用</Typography.Text>
</Space>
</div>
<Select
style={{marginTop: 8}}
placeholder={'请选择该渠道所支持的模型'}
name='models'
required
multiple
selection
onChange={value => {
handleInputChange('model_limits', value);
}}
value={inputs.model_limits}
autoComplete='new-password'
optionList={models}
disabled={!model_limits_enabled}
/>
</Spin>
</SideSheet>
</>
);
}; };
export default EditToken; export default EditToken;

View File

@ -1,98 +1,98 @@
import React, {useState} from 'react'; import React, { useState } from 'react';
import {API, isMobile, showError, showSuccess} from '../../helpers'; import { API, isMobile, showError, showSuccess } from '../../helpers';
import Title from "@douyinfe/semi-ui/lib/es/typography/title"; import Title from '@douyinfe/semi-ui/lib/es/typography/title';
import {Button, SideSheet, Space, Input, Spin} from "@douyinfe/semi-ui"; import { Button, Input, SideSheet, Space, Spin } from '@douyinfe/semi-ui';
const AddUser = (props) => { const AddUser = (props) => {
const originInputs = { const originInputs = {
username: '', username: '',
display_name: '', display_name: '',
password: '', password: ''
}; };
const [inputs, setInputs] = useState(originInputs); const [inputs, setInputs] = useState(originInputs);
const [loading, setLoading] = useState(false); const [loading, setLoading] = useState(false);
const {username, display_name, password} = inputs; const { username, display_name, password } = inputs;
const handleInputChange = (name, value) => { const handleInputChange = (name, value) => {
setInputs((inputs) => ({...inputs, [name]: value})); setInputs((inputs) => ({ ...inputs, [name]: value }));
}; };
const submit = async () => { const submit = async () => {
setLoading(true); setLoading(true);
if (inputs.username === '' || inputs.password === '') return; if (inputs.username === '' || inputs.password === '') return;
const res = await API.post(`/api/user/`, inputs); const res = await API.post(`/api/user/`, inputs);
const {success, message} = res.data; const { success, message } = res.data;
if (success) { if (success) {
showSuccess('用户账户创建成功!'); showSuccess('用户账户创建成功!');
setInputs(originInputs); setInputs(originInputs);
props.refresh(); props.refresh();
props.handleClose(); props.handleClose();
} else { } else {
showError(message); showError(message);
}
setLoading(false);
};
const handleCancel = () => {
props.handleClose();
} }
setLoading(false);
};
return ( const handleCancel = () => {
<> props.handleClose();
<SideSheet };
placement={'left'}
title={<Title level={3}>{'添加用户'}</Title>} return (
headerStyle={{borderBottom: '1px solid var(--semi-color-border)'}} <>
bodyStyle={{borderBottom: '1px solid var(--semi-color-border)'}} <SideSheet
visible={props.visible} placement={'left'}
footer={ title={<Title level={3}>{'添加用户'}</Title>}
<div style={{display: 'flex', justifyContent: 'flex-end'}}> headerStyle={{ borderBottom: '1px solid var(--semi-color-border)' }}
<Space> bodyStyle={{ borderBottom: '1px solid var(--semi-color-border)' }}
<Button theme='solid' size={'large'} onClick={submit}>提交</Button> visible={props.visible}
<Button theme='solid' size={'large'} type={'tertiary'} onClick={handleCancel}>取消</Button> footer={
</Space> <div style={{ display: 'flex', justifyContent: 'flex-end' }}>
</div> <Space>
} <Button theme="solid" size={'large'} onClick={submit}>提交</Button>
closeIcon={null} <Button theme="solid" size={'large'} type={'tertiary'} onClick={handleCancel}>取消</Button>
onCancel={() => handleCancel()} </Space>
width={isMobile() ? '100%' : 600} </div>
> }
<Spin spinning={loading}> closeIcon={null}
<Input onCancel={() => handleCancel()}
style={{marginTop: 20}} width={isMobile() ? '100%' : 600}
label="用户名" >
name="username" <Spin spinning={loading}>
addonBefore={'用户名'} <Input
placeholder={'请输入用户名'} style={{ marginTop: 20 }}
onChange={value => handleInputChange('username', value)} label="用户名"
value={username} name="username"
autoComplete="off" addonBefore={'用户名'}
/> placeholder={'请输入用户名'}
<Input onChange={value => handleInputChange('username', value)}
style={{marginTop: 20}} value={username}
addonBefore={'显示名'} autoComplete="off"
label="显示名称" />
name="display_name" <Input
autoComplete="off" style={{ marginTop: 20 }}
placeholder={'请输入显示名称'} addonBefore={'显示名'}
onChange={value => handleInputChange('display_name', value)} label="显示名称"
value={display_name} name="display_name"
/> autoComplete="off"
<Input placeholder={'请输入显示名称'}
style={{marginTop: 20}} onChange={value => handleInputChange('display_name', value)}
label="密 码" value={display_name}
name="password" />
type={'password'} <Input
addonBefore={'密码'} style={{ marginTop: 20 }}
placeholder={'请输入密码'} label="密 码"
onChange={value => handleInputChange('password', value)} name="password"
value={password} type={'password'}
autoComplete="off" addonBefore={'密码'}
/> placeholder={'请输入密码'}
</Spin> onChange={value => handleInputChange('password', value)}
</SideSheet> value={password}
</> autoComplete="off"
); />
</Spin>
</SideSheet>
</>
);
}; };
export default AddUser; export default AddUser;

View File

@ -1,9 +1,9 @@
import React, { useEffect, useState } from 'react'; import React, { useEffect, useState } from 'react';
import { useParams, useNavigate } from 'react-router-dom'; import { useNavigate } from 'react-router-dom';
import { API, isMobile, showError, showSuccess } from '../../helpers'; import { API, isMobile, showError, showSuccess } from '../../helpers';
import { renderQuota, renderQuotaWithPrompt } from '../../helpers/render'; import { renderQuotaWithPrompt } from '../../helpers/render';
import Title from "@douyinfe/semi-ui/lib/es/typography/title"; import Title from '@douyinfe/semi-ui/lib/es/typography/title';
import { SideSheet, Space, Button, Spin, Input, Typography, Select, Divider } from "@douyinfe/semi-ui"; import { Button, Divider, Input, Select, SideSheet, Space, Spin, Typography } from '@douyinfe/semi-ui';
const EditUser = (props) => { const EditUser = (props) => {
const userId = props.editingUser.id; const userId = props.editingUser.id;
@ -29,7 +29,7 @@ const EditUser = (props) => {
let res = await API.get(`/api/group/`); let res = await API.get(`/api/group/`);
setGroupOptions(res.data.data.map((group) => ({ setGroupOptions(res.data.data.map((group) => ({
label: group, label: group,
value: group, value: group
}))); })));
} catch (error) { } catch (error) {
showError(error.message); showError(error.message);
@ -38,7 +38,7 @@ const EditUser = (props) => {
const navigate = useNavigate(); const navigate = useNavigate();
const handleCancel = () => { const handleCancel = () => {
props.handleClose(); props.handleClose();
} };
const loadUser = async () => { const loadUser = async () => {
setLoading(true); setLoading(true);
let res = undefined; let res = undefined;
@ -98,8 +98,8 @@ const EditUser = (props) => {
footer={ footer={
<div style={{ display: 'flex', justifyContent: 'flex-end' }}> <div style={{ display: 'flex', justifyContent: 'flex-end' }}>
<Space> <Space>
<Button theme='solid' size={'large'} onClick={submit}>提交</Button> <Button theme="solid" size={'large'} onClick={submit}>提交</Button>
<Button theme='solid' size={'large'} type={'tertiary'} onClick={handleCancel}>取消</Button> <Button theme="solid" size={'large'} type={'tertiary'} onClick={handleCancel}>取消</Button>
</Space> </Space>
</div> </div>
} }
@ -112,35 +112,35 @@ const EditUser = (props) => {
<Typography.Text>用户名</Typography.Text> <Typography.Text>用户名</Typography.Text>
</div> </div>
<Input <Input
label='用户名' label="用户名"
name='username' name="username"
placeholder={'请输入新的用户名'} placeholder={'请输入新的用户名'}
onChange={value => handleInputChange('username', value)} onChange={value => handleInputChange('username', value)}
value={username} value={username}
autoComplete='new-password' autoComplete="new-password"
/> />
<div style={{ marginTop: 20 }}> <div style={{ marginTop: 20 }}>
<Typography.Text>密码</Typography.Text> <Typography.Text>密码</Typography.Text>
</div> </div>
<Input <Input
label='密码' label="密码"
name='password' name="password"
type={'password'} type={'password'}
placeholder={'请输入新的密码,最短 8 位'} placeholder={'请输入新的密码,最短 8 位'}
onChange={value => handleInputChange('password', value)} onChange={value => handleInputChange('password', value)}
value={password} value={password}
autoComplete='new-password' autoComplete="new-password"
/> />
<div style={{ marginTop: 20 }}> <div style={{ marginTop: 20 }}>
<Typography.Text>显示名称</Typography.Text> <Typography.Text>显示名称</Typography.Text>
</div> </div>
<Input <Input
label='显示名称' label="显示名称"
name='display_name' name="display_name"
placeholder={'请输入新的显示名称'} placeholder={'请输入新的显示名称'}
onChange={value => handleInputChange('display_name', value)} onChange={value => handleInputChange('display_name', value)}
value={display_name} value={display_name}
autoComplete='new-password' autoComplete="new-password"
/> />
{ {
userId && <> userId && <>
@ -149,7 +149,7 @@ const EditUser = (props) => {
</div> </div>
<Select <Select
placeholder={'请选择分组'} placeholder={'请选择分组'}
name='group' name="group"
fluid fluid
search search
selection selection
@ -157,19 +157,19 @@ const EditUser = (props) => {
additionLabel={'请在系统设置页面编辑分组倍率以添加新的分组:'} additionLabel={'请在系统设置页面编辑分组倍率以添加新的分组:'}
onChange={value => handleInputChange('group', value)} onChange={value => handleInputChange('group', value)}
value={inputs.group} value={inputs.group}
autoComplete='new-password' autoComplete="new-password"
optionList={groupOptions} optionList={groupOptions}
/> />
<div style={{ marginTop: 20 }}> <div style={{ marginTop: 20 }}>
<Typography.Text>{`剩余额度${renderQuotaWithPrompt(quota)}`}</Typography.Text> <Typography.Text>{`剩余额度${renderQuotaWithPrompt(quota)}`}</Typography.Text>
</div> </div>
<Input <Input
name='quota' name="quota"
placeholder={'请输入新的剩余额度'} placeholder={'请输入新的剩余额度'}
onChange={value => handleInputChange('quota', value)} onChange={value => handleInputChange('quota', value)}
value={quota} value={quota}
type={'number'} type={'number'}
autoComplete='new-password' autoComplete="new-password"
/> />
</> </>
} }
@ -178,37 +178,37 @@ const EditUser = (props) => {
<Typography.Text>已绑定的 GitHub 账户</Typography.Text> <Typography.Text>已绑定的 GitHub 账户</Typography.Text>
</div> </div>
<Input <Input
name='github_id' name="github_id"
value={github_id} value={github_id}
autoComplete='new-password' autoComplete="new-password"
placeholder='此项只读,需要用户通过个人设置页面的相关绑定按钮进行绑定,不可直接修改' placeholder="此项只读,需要用户通过个人设置页面的相关绑定按钮进行绑定,不可直接修改"
readonly readonly
/> />
<div style={{ marginTop: 20 }}> <div style={{ marginTop: 20 }}>
<Typography.Text>已绑定的微信账户</Typography.Text> <Typography.Text>已绑定的微信账户</Typography.Text>
</div> </div>
<Input <Input
name='wechat_id' name="wechat_id"
value={wechat_id} value={wechat_id}
autoComplete='new-password' autoComplete="new-password"
placeholder='此项只读,需要用户通过个人设置页面的相关绑定按钮进行绑定,不可直接修改' placeholder="此项只读,需要用户通过个人设置页面的相关绑定按钮进行绑定,不可直接修改"
readonly readonly
/> />
<Input <Input
name='telegram_id' name="telegram_id"
value={telegram_id} value={telegram_id}
autoComplete='new-password' autoComplete="new-password"
placeholder='此项只读,需要用户通过个人设置页面的相关绑定按钮进行绑定,不可直接修改' placeholder="此项只读,需要用户通过个人设置页面的相关绑定按钮进行绑定,不可直接修改"
readonly readonly
/> />
<div style={{ marginTop: 20 }}> <div style={{ marginTop: 20 }}>
<Typography.Text>已绑定的邮箱账户</Typography.Text> <Typography.Text>已绑定的邮箱账户</Typography.Text>
</div> </div>
<Input <Input
name='email' name="email"
value={email} value={email}
autoComplete='new-password' autoComplete="new-password"
placeholder='此项只读,需要用户通过个人设置页面的相关绑定按钮进行绑定,不可直接修改' placeholder="此项只读,需要用户通过个人设置页面的相关绑定按钮进行绑定,不可直接修改"
readonly readonly
/> />
</Spin> </Spin>