2025-07-19 03:30:44 +08:00
|
|
|
|
/*
|
|
|
|
|
|
Copyright (C) 2025 QuantumNous
|
|
|
|
|
|
|
|
|
|
|
|
This program is free software: you can redistribute it and/or modify
|
|
|
|
|
|
it under the terms of the GNU Affero General Public License as
|
|
|
|
|
|
published by the Free Software Foundation, either version 3 of the
|
|
|
|
|
|
License, or (at your option) any later version.
|
|
|
|
|
|
|
|
|
|
|
|
This program is distributed in the hope that it will be useful,
|
|
|
|
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
|
|
GNU Affero General Public License for more details.
|
|
|
|
|
|
|
|
|
|
|
|
You should have received a copy of the GNU Affero General Public License
|
|
|
|
|
|
along with this program. If not, see <https://www.gnu.org/licenses/>.
|
|
|
|
|
|
|
|
|
|
|
|
For commercial licensing, please contact support@quantumnous.com
|
|
|
|
|
|
*/
|
|
|
|
|
|
|
2025-07-18 22:04:54 +08:00
|
|
|
|
import React from 'react';
|
|
|
|
|
|
import {
|
|
|
|
|
|
Avatar,
|
|
|
|
|
|
Space,
|
|
|
|
|
|
Tag,
|
|
|
|
|
|
Tooltip,
|
|
|
|
|
|
Popover,
|
2025-08-30 21:15:10 +08:00
|
|
|
|
Typography,
|
2025-07-18 22:04:54 +08:00
|
|
|
|
} from '@douyinfe/semi-ui';
|
|
|
|
|
|
import {
|
|
|
|
|
|
timestamp2string,
|
|
|
|
|
|
renderGroup,
|
|
|
|
|
|
renderQuota,
|
|
|
|
|
|
stringToColor,
|
|
|
|
|
|
getLogOther,
|
|
|
|
|
|
renderModelTag,
|
|
|
|
|
|
renderClaudeLogContent,
|
|
|
|
|
|
renderLogContent,
|
|
|
|
|
|
renderModelPriceSimple,
|
|
|
|
|
|
renderAudioModelPrice,
|
|
|
|
|
|
renderClaudeModelPrice,
|
2025-08-30 21:15:10 +08:00
|
|
|
|
renderModelPrice,
|
2025-07-18 22:04:54 +08:00
|
|
|
|
} from '../../../helpers';
|
2026-01-26 19:57:41 +08:00
|
|
|
|
import { IconHelpCircle, IconStarStroked } from '@douyinfe/semi-icons';
|
2025-07-18 22:04:54 +08:00
|
|
|
|
import { Route } from 'lucide-react';
|
|
|
|
|
|
|
|
|
|
|
|
const colors = [
|
|
|
|
|
|
'amber',
|
|
|
|
|
|
'blue',
|
|
|
|
|
|
'cyan',
|
|
|
|
|
|
'green',
|
|
|
|
|
|
'grey',
|
|
|
|
|
|
'indigo',
|
|
|
|
|
|
'light-blue',
|
|
|
|
|
|
'lime',
|
|
|
|
|
|
'orange',
|
|
|
|
|
|
'pink',
|
|
|
|
|
|
'purple',
|
|
|
|
|
|
'red',
|
|
|
|
|
|
'teal',
|
|
|
|
|
|
'violet',
|
|
|
|
|
|
'yellow',
|
|
|
|
|
|
];
|
|
|
|
|
|
|
2026-01-26 20:20:30 +08:00
|
|
|
|
function formatRatio(ratio) {
|
|
|
|
|
|
if (ratio === undefined || ratio === null) {
|
|
|
|
|
|
return '-';
|
|
|
|
|
|
}
|
|
|
|
|
|
if (typeof ratio === 'number') {
|
|
|
|
|
|
return ratio.toFixed(4);
|
|
|
|
|
|
}
|
|
|
|
|
|
return String(ratio);
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2025-07-18 22:04:54 +08:00
|
|
|
|
// Render functions
|
|
|
|
|
|
function renderType(type, t) {
|
|
|
|
|
|
switch (type) {
|
|
|
|
|
|
case 1:
|
|
|
|
|
|
return (
|
|
|
|
|
|
<Tag color='cyan' shape='circle'>
|
|
|
|
|
|
{t('充值')}
|
|
|
|
|
|
</Tag>
|
|
|
|
|
|
);
|
|
|
|
|
|
case 2:
|
|
|
|
|
|
return (
|
|
|
|
|
|
<Tag color='lime' shape='circle'>
|
|
|
|
|
|
{t('消费')}
|
|
|
|
|
|
</Tag>
|
|
|
|
|
|
);
|
|
|
|
|
|
case 3:
|
|
|
|
|
|
return (
|
|
|
|
|
|
<Tag color='orange' shape='circle'>
|
|
|
|
|
|
{t('管理')}
|
|
|
|
|
|
</Tag>
|
|
|
|
|
|
);
|
|
|
|
|
|
case 4:
|
|
|
|
|
|
return (
|
|
|
|
|
|
<Tag color='purple' shape='circle'>
|
|
|
|
|
|
{t('系统')}
|
|
|
|
|
|
</Tag>
|
|
|
|
|
|
);
|
|
|
|
|
|
case 5:
|
|
|
|
|
|
return (
|
|
|
|
|
|
<Tag color='red' shape='circle'>
|
|
|
|
|
|
{t('错误')}
|
|
|
|
|
|
</Tag>
|
|
|
|
|
|
);
|
|
|
|
|
|
default:
|
|
|
|
|
|
return (
|
|
|
|
|
|
<Tag color='grey' shape='circle'>
|
|
|
|
|
|
{t('未知')}
|
|
|
|
|
|
</Tag>
|
|
|
|
|
|
);
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
function renderIsStream(bool, t) {
|
|
|
|
|
|
if (bool) {
|
|
|
|
|
|
return (
|
|
|
|
|
|
<Tag color='blue' shape='circle'>
|
|
|
|
|
|
{t('流')}
|
|
|
|
|
|
</Tag>
|
|
|
|
|
|
);
|
|
|
|
|
|
} else {
|
|
|
|
|
|
return (
|
|
|
|
|
|
<Tag color='purple' shape='circle'>
|
|
|
|
|
|
{t('非流')}
|
|
|
|
|
|
</Tag>
|
|
|
|
|
|
);
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
function renderUseTime(type, t) {
|
|
|
|
|
|
const time = parseInt(type);
|
|
|
|
|
|
if (time < 101) {
|
|
|
|
|
|
return (
|
|
|
|
|
|
<Tag color='green' shape='circle'>
|
|
|
|
|
|
{' '}
|
|
|
|
|
|
{time} s{' '}
|
|
|
|
|
|
</Tag>
|
|
|
|
|
|
);
|
|
|
|
|
|
} else if (time < 300) {
|
|
|
|
|
|
return (
|
|
|
|
|
|
<Tag color='orange' shape='circle'>
|
|
|
|
|
|
{' '}
|
|
|
|
|
|
{time} s{' '}
|
|
|
|
|
|
</Tag>
|
|
|
|
|
|
);
|
|
|
|
|
|
} else {
|
|
|
|
|
|
return (
|
|
|
|
|
|
<Tag color='red' shape='circle'>
|
|
|
|
|
|
{' '}
|
|
|
|
|
|
{time} s{' '}
|
|
|
|
|
|
</Tag>
|
|
|
|
|
|
);
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
function renderFirstUseTime(type, t) {
|
|
|
|
|
|
let time = parseFloat(type) / 1000.0;
|
|
|
|
|
|
time = time.toFixed(1);
|
|
|
|
|
|
if (time < 3) {
|
|
|
|
|
|
return (
|
|
|
|
|
|
<Tag color='green' shape='circle'>
|
|
|
|
|
|
{' '}
|
|
|
|
|
|
{time} s{' '}
|
|
|
|
|
|
</Tag>
|
|
|
|
|
|
);
|
|
|
|
|
|
} else if (time < 10) {
|
|
|
|
|
|
return (
|
|
|
|
|
|
<Tag color='orange' shape='circle'>
|
|
|
|
|
|
{' '}
|
|
|
|
|
|
{time} s{' '}
|
|
|
|
|
|
</Tag>
|
|
|
|
|
|
);
|
|
|
|
|
|
} else {
|
|
|
|
|
|
return (
|
|
|
|
|
|
<Tag color='red' shape='circle'>
|
|
|
|
|
|
{' '}
|
|
|
|
|
|
{time} s{' '}
|
|
|
|
|
|
</Tag>
|
|
|
|
|
|
);
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
function renderModelName(record, copyText, t) {
|
|
|
|
|
|
let other = getLogOther(record.other);
|
|
|
|
|
|
let modelMapped =
|
|
|
|
|
|
other?.is_model_mapped &&
|
|
|
|
|
|
other?.upstream_model_name &&
|
|
|
|
|
|
other?.upstream_model_name !== '';
|
|
|
|
|
|
if (!modelMapped) {
|
|
|
|
|
|
return renderModelTag(record.model_name, {
|
|
|
|
|
|
onClick: (event) => {
|
2025-08-30 21:15:10 +08:00
|
|
|
|
copyText(event, record.model_name).then((r) => {});
|
2025-07-18 22:04:54 +08:00
|
|
|
|
},
|
|
|
|
|
|
});
|
|
|
|
|
|
} else {
|
|
|
|
|
|
return (
|
|
|
|
|
|
<>
|
|
|
|
|
|
<Space vertical align={'start'}>
|
|
|
|
|
|
<Popover
|
|
|
|
|
|
content={
|
|
|
|
|
|
<div style={{ padding: 10 }}>
|
|
|
|
|
|
<Space vertical align={'start'}>
|
|
|
|
|
|
<div className='flex items-center'>
|
|
|
|
|
|
<Typography.Text strong style={{ marginRight: 8 }}>
|
|
|
|
|
|
{t('请求并计费模型')}:
|
|
|
|
|
|
</Typography.Text>
|
|
|
|
|
|
{renderModelTag(record.model_name, {
|
|
|
|
|
|
onClick: (event) => {
|
2025-08-30 21:15:10 +08:00
|
|
|
|
copyText(event, record.model_name).then((r) => {});
|
2025-07-18 22:04:54 +08:00
|
|
|
|
},
|
|
|
|
|
|
})}
|
|
|
|
|
|
</div>
|
|
|
|
|
|
<div className='flex items-center'>
|
|
|
|
|
|
<Typography.Text strong style={{ marginRight: 8 }}>
|
|
|
|
|
|
{t('实际模型')}:
|
|
|
|
|
|
</Typography.Text>
|
|
|
|
|
|
{renderModelTag(other.upstream_model_name, {
|
|
|
|
|
|
onClick: (event) => {
|
|
|
|
|
|
copyText(event, other.upstream_model_name).then(
|
2025-08-30 21:15:10 +08:00
|
|
|
|
(r) => {},
|
2025-07-18 22:04:54 +08:00
|
|
|
|
);
|
|
|
|
|
|
},
|
|
|
|
|
|
})}
|
|
|
|
|
|
</div>
|
|
|
|
|
|
</Space>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
}
|
|
|
|
|
|
>
|
|
|
|
|
|
{renderModelTag(record.model_name, {
|
|
|
|
|
|
onClick: (event) => {
|
2025-08-30 21:15:10 +08:00
|
|
|
|
copyText(event, record.model_name).then((r) => {});
|
2025-07-18 22:04:54 +08:00
|
|
|
|
},
|
|
|
|
|
|
suffixIcon: (
|
|
|
|
|
|
<Route
|
|
|
|
|
|
style={{ width: '0.9em', height: '0.9em', opacity: 0.75 }}
|
|
|
|
|
|
/>
|
|
|
|
|
|
),
|
|
|
|
|
|
})}
|
|
|
|
|
|
</Popover>
|
|
|
|
|
|
</Space>
|
|
|
|
|
|
</>
|
|
|
|
|
|
);
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
export const getLogsColumns = ({
|
|
|
|
|
|
t,
|
|
|
|
|
|
COLUMN_KEYS,
|
|
|
|
|
|
copyText,
|
|
|
|
|
|
showUserInfoFunc,
|
|
|
|
|
|
isAdminUser,
|
|
|
|
|
|
}) => {
|
|
|
|
|
|
return [
|
|
|
|
|
|
{
|
|
|
|
|
|
key: COLUMN_KEYS.TIME,
|
|
|
|
|
|
title: t('时间'),
|
|
|
|
|
|
dataIndex: 'timestamp2string',
|
|
|
|
|
|
},
|
|
|
|
|
|
{
|
|
|
|
|
|
key: COLUMN_KEYS.CHANNEL,
|
|
|
|
|
|
title: t('渠道'),
|
|
|
|
|
|
dataIndex: 'channel',
|
|
|
|
|
|
render: (text, record, index) => {
|
|
|
|
|
|
let isMultiKey = false;
|
|
|
|
|
|
let multiKeyIndex = -1;
|
|
|
|
|
|
let other = getLogOther(record.other);
|
|
|
|
|
|
if (other?.admin_info) {
|
|
|
|
|
|
let adminInfo = other.admin_info;
|
|
|
|
|
|
if (adminInfo?.is_multi_key) {
|
|
|
|
|
|
isMultiKey = true;
|
|
|
|
|
|
multiKeyIndex = adminInfo.multi_key_index;
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2025-08-30 21:15:10 +08:00
|
|
|
|
return isAdminUser &&
|
|
|
|
|
|
(record.type === 0 || record.type === 2 || record.type === 5) ? (
|
2025-07-18 22:04:54 +08:00
|
|
|
|
<Space>
|
|
|
|
|
|
<Tooltip content={record.channel_name || t('未知渠道')}>
|
2025-07-19 15:05:31 +08:00
|
|
|
|
<span>
|
|
|
|
|
|
<Tag
|
|
|
|
|
|
color={colors[parseInt(text) % colors.length]}
|
|
|
|
|
|
shape='circle'
|
|
|
|
|
|
>
|
|
|
|
|
|
{text}
|
|
|
|
|
|
</Tag>
|
|
|
|
|
|
</span>
|
2025-07-18 22:04:54 +08:00
|
|
|
|
</Tooltip>
|
|
|
|
|
|
{isMultiKey && (
|
|
|
|
|
|
<Tag color='white' shape='circle'>
|
|
|
|
|
|
{multiKeyIndex}
|
|
|
|
|
|
</Tag>
|
|
|
|
|
|
)}
|
|
|
|
|
|
</Space>
|
|
|
|
|
|
) : null;
|
|
|
|
|
|
},
|
|
|
|
|
|
},
|
|
|
|
|
|
{
|
|
|
|
|
|
key: COLUMN_KEYS.USERNAME,
|
|
|
|
|
|
title: t('用户'),
|
|
|
|
|
|
dataIndex: 'username',
|
|
|
|
|
|
render: (text, record, index) => {
|
|
|
|
|
|
return isAdminUser ? (
|
|
|
|
|
|
<div>
|
|
|
|
|
|
<Avatar
|
|
|
|
|
|
size='extra-small'
|
|
|
|
|
|
color={stringToColor(text)}
|
|
|
|
|
|
style={{ marginRight: 4 }}
|
|
|
|
|
|
onClick={(event) => {
|
|
|
|
|
|
event.stopPropagation();
|
|
|
|
|
|
showUserInfoFunc(record.user_id);
|
|
|
|
|
|
}}
|
|
|
|
|
|
>
|
|
|
|
|
|
{typeof text === 'string' && text.slice(0, 1)}
|
|
|
|
|
|
</Avatar>
|
|
|
|
|
|
{text}
|
|
|
|
|
|
</div>
|
|
|
|
|
|
) : (
|
|
|
|
|
|
<></>
|
|
|
|
|
|
);
|
|
|
|
|
|
},
|
|
|
|
|
|
},
|
|
|
|
|
|
{
|
|
|
|
|
|
key: COLUMN_KEYS.TOKEN,
|
|
|
|
|
|
title: t('令牌'),
|
|
|
|
|
|
dataIndex: 'token_name',
|
|
|
|
|
|
render: (text, record, index) => {
|
|
|
|
|
|
return record.type === 0 || record.type === 2 || record.type === 5 ? (
|
|
|
|
|
|
<div>
|
|
|
|
|
|
<Tag
|
|
|
|
|
|
color='grey'
|
|
|
|
|
|
shape='circle'
|
|
|
|
|
|
onClick={(event) => {
|
|
|
|
|
|
copyText(event, text);
|
|
|
|
|
|
}}
|
|
|
|
|
|
>
|
|
|
|
|
|
{' '}
|
|
|
|
|
|
{t(text)}{' '}
|
|
|
|
|
|
</Tag>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
) : (
|
|
|
|
|
|
<></>
|
|
|
|
|
|
);
|
|
|
|
|
|
},
|
|
|
|
|
|
},
|
|
|
|
|
|
{
|
|
|
|
|
|
key: COLUMN_KEYS.GROUP,
|
|
|
|
|
|
title: t('分组'),
|
|
|
|
|
|
dataIndex: 'group',
|
|
|
|
|
|
render: (text, record, index) => {
|
|
|
|
|
|
if (record.type === 0 || record.type === 2 || record.type === 5) {
|
|
|
|
|
|
if (record.group) {
|
|
|
|
|
|
return <>{renderGroup(record.group)}</>;
|
|
|
|
|
|
} else {
|
|
|
|
|
|
let other = null;
|
|
|
|
|
|
try {
|
|
|
|
|
|
other = JSON.parse(record.other);
|
|
|
|
|
|
} catch (e) {
|
|
|
|
|
|
console.error(
|
|
|
|
|
|
`Failed to parse record.other: "${record.other}".`,
|
|
|
|
|
|
e,
|
|
|
|
|
|
);
|
|
|
|
|
|
}
|
|
|
|
|
|
if (other === null) {
|
|
|
|
|
|
return <></>;
|
|
|
|
|
|
}
|
|
|
|
|
|
if (other.group !== undefined) {
|
|
|
|
|
|
return <>{renderGroup(other.group)}</>;
|
|
|
|
|
|
} else {
|
|
|
|
|
|
return <></>;
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
} else {
|
|
|
|
|
|
return <></>;
|
|
|
|
|
|
}
|
|
|
|
|
|
},
|
|
|
|
|
|
},
|
|
|
|
|
|
{
|
|
|
|
|
|
key: COLUMN_KEYS.TYPE,
|
|
|
|
|
|
title: t('类型'),
|
|
|
|
|
|
dataIndex: 'type',
|
|
|
|
|
|
render: (text, record, index) => {
|
|
|
|
|
|
return <>{renderType(text, t)}</>;
|
|
|
|
|
|
},
|
|
|
|
|
|
},
|
|
|
|
|
|
{
|
|
|
|
|
|
key: COLUMN_KEYS.MODEL,
|
|
|
|
|
|
title: t('模型'),
|
|
|
|
|
|
dataIndex: 'model_name',
|
|
|
|
|
|
render: (text, record, index) => {
|
|
|
|
|
|
return record.type === 0 || record.type === 2 || record.type === 5 ? (
|
|
|
|
|
|
<>{renderModelName(record, copyText, t)}</>
|
|
|
|
|
|
) : (
|
|
|
|
|
|
<></>
|
|
|
|
|
|
);
|
|
|
|
|
|
},
|
|
|
|
|
|
},
|
|
|
|
|
|
{
|
|
|
|
|
|
key: COLUMN_KEYS.USE_TIME,
|
|
|
|
|
|
title: t('用时/首字'),
|
|
|
|
|
|
dataIndex: 'use_time',
|
|
|
|
|
|
render: (text, record, index) => {
|
|
|
|
|
|
if (!(record.type === 2 || record.type === 5)) {
|
|
|
|
|
|
return <></>;
|
|
|
|
|
|
}
|
|
|
|
|
|
if (record.is_stream) {
|
|
|
|
|
|
let other = getLogOther(record.other);
|
|
|
|
|
|
return (
|
|
|
|
|
|
<>
|
|
|
|
|
|
<Space>
|
|
|
|
|
|
{renderUseTime(text, t)}
|
|
|
|
|
|
{renderFirstUseTime(other?.frt, t)}
|
|
|
|
|
|
{renderIsStream(record.is_stream, t)}
|
|
|
|
|
|
</Space>
|
|
|
|
|
|
</>
|
|
|
|
|
|
);
|
|
|
|
|
|
} else {
|
|
|
|
|
|
return (
|
|
|
|
|
|
<>
|
|
|
|
|
|
<Space>
|
|
|
|
|
|
{renderUseTime(text, t)}
|
|
|
|
|
|
{renderIsStream(record.is_stream, t)}
|
|
|
|
|
|
</Space>
|
|
|
|
|
|
</>
|
|
|
|
|
|
);
|
|
|
|
|
|
}
|
|
|
|
|
|
},
|
|
|
|
|
|
},
|
|
|
|
|
|
{
|
|
|
|
|
|
key: COLUMN_KEYS.PROMPT,
|
2025-10-03 21:49:24 +08:00
|
|
|
|
title: t('输入'),
|
2025-07-18 22:04:54 +08:00
|
|
|
|
dataIndex: 'prompt_tokens',
|
|
|
|
|
|
render: (text, record, index) => {
|
|
|
|
|
|
return record.type === 0 || record.type === 2 || record.type === 5 ? (
|
|
|
|
|
|
<>{<span> {text} </span>}</>
|
|
|
|
|
|
) : (
|
|
|
|
|
|
<></>
|
|
|
|
|
|
);
|
|
|
|
|
|
},
|
|
|
|
|
|
},
|
|
|
|
|
|
{
|
|
|
|
|
|
key: COLUMN_KEYS.COMPLETION,
|
2025-10-03 21:49:24 +08:00
|
|
|
|
title: t('输出'),
|
2025-07-18 22:04:54 +08:00
|
|
|
|
dataIndex: 'completion_tokens',
|
|
|
|
|
|
render: (text, record, index) => {
|
|
|
|
|
|
return parseInt(text) > 0 &&
|
|
|
|
|
|
(record.type === 0 || record.type === 2 || record.type === 5) ? (
|
|
|
|
|
|
<>{<span> {text} </span>}</>
|
|
|
|
|
|
) : (
|
|
|
|
|
|
<></>
|
|
|
|
|
|
);
|
|
|
|
|
|
},
|
|
|
|
|
|
},
|
|
|
|
|
|
{
|
|
|
|
|
|
key: COLUMN_KEYS.COST,
|
|
|
|
|
|
title: t('花费'),
|
|
|
|
|
|
dataIndex: 'quota',
|
|
|
|
|
|
render: (text, record, index) => {
|
|
|
|
|
|
return record.type === 0 || record.type === 2 || record.type === 5 ? (
|
|
|
|
|
|
<>{renderQuota(text, 6)}</>
|
|
|
|
|
|
) : (
|
|
|
|
|
|
<></>
|
|
|
|
|
|
);
|
|
|
|
|
|
},
|
|
|
|
|
|
},
|
|
|
|
|
|
{
|
|
|
|
|
|
key: COLUMN_KEYS.IP,
|
|
|
|
|
|
title: (
|
2025-08-30 21:15:10 +08:00
|
|
|
|
<div className='flex items-center gap-1'>
|
2025-07-18 22:04:54 +08:00
|
|
|
|
{t('IP')}
|
2025-08-30 21:15:10 +08:00
|
|
|
|
<Tooltip
|
|
|
|
|
|
content={t(
|
|
|
|
|
|
'只有当用户设置开启IP记录时,才会进行请求和错误类型日志的IP记录',
|
|
|
|
|
|
)}
|
|
|
|
|
|
>
|
|
|
|
|
|
<IconHelpCircle className='text-gray-400 cursor-help' />
|
2025-07-18 22:04:54 +08:00
|
|
|
|
</Tooltip>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
),
|
|
|
|
|
|
dataIndex: 'ip',
|
|
|
|
|
|
render: (text, record, index) => {
|
|
|
|
|
|
return (record.type === 2 || record.type === 5) && text ? (
|
|
|
|
|
|
<Tooltip content={text}>
|
2025-07-19 15:05:31 +08:00
|
|
|
|
<span>
|
|
|
|
|
|
<Tag
|
|
|
|
|
|
color='orange'
|
|
|
|
|
|
shape='circle'
|
|
|
|
|
|
onClick={(event) => {
|
|
|
|
|
|
copyText(event, text);
|
|
|
|
|
|
}}
|
|
|
|
|
|
>
|
|
|
|
|
|
{text}
|
|
|
|
|
|
</Tag>
|
|
|
|
|
|
</span>
|
2025-07-18 22:04:54 +08:00
|
|
|
|
</Tooltip>
|
|
|
|
|
|
) : (
|
|
|
|
|
|
<></>
|
|
|
|
|
|
);
|
|
|
|
|
|
},
|
|
|
|
|
|
},
|
|
|
|
|
|
{
|
|
|
|
|
|
key: COLUMN_KEYS.RETRY,
|
|
|
|
|
|
title: t('重试'),
|
|
|
|
|
|
dataIndex: 'retry',
|
|
|
|
|
|
render: (text, record, index) => {
|
|
|
|
|
|
if (!(record.type === 2 || record.type === 5)) {
|
|
|
|
|
|
return <></>;
|
|
|
|
|
|
}
|
|
|
|
|
|
let content = t('渠道') + `:${record.channel}`;
|
2026-01-26 19:57:41 +08:00
|
|
|
|
let affinity = null;
|
2025-07-18 22:04:54 +08:00
|
|
|
|
if (record.other !== '') {
|
|
|
|
|
|
let other = JSON.parse(record.other);
|
|
|
|
|
|
if (other === null) {
|
|
|
|
|
|
return <></>;
|
|
|
|
|
|
}
|
|
|
|
|
|
if (other.admin_info !== undefined) {
|
|
|
|
|
|
if (
|
|
|
|
|
|
other.admin_info.use_channel !== null &&
|
|
|
|
|
|
other.admin_info.use_channel !== undefined &&
|
|
|
|
|
|
other.admin_info.use_channel !== ''
|
|
|
|
|
|
) {
|
|
|
|
|
|
let useChannel = other.admin_info.use_channel;
|
|
|
|
|
|
let useChannelStr = useChannel.join('->');
|
|
|
|
|
|
content = t('渠道') + `:${useChannelStr}`;
|
|
|
|
|
|
}
|
2026-01-26 19:57:41 +08:00
|
|
|
|
if (other.admin_info.channel_affinity) {
|
|
|
|
|
|
affinity = other.admin_info.channel_affinity;
|
|
|
|
|
|
}
|
2025-07-18 22:04:54 +08:00
|
|
|
|
}
|
|
|
|
|
|
}
|
2026-01-26 19:57:41 +08:00
|
|
|
|
return isAdminUser ? (
|
|
|
|
|
|
<Space>
|
|
|
|
|
|
<div>{content}</div>
|
|
|
|
|
|
{affinity ? (
|
|
|
|
|
|
<Tooltip
|
|
|
|
|
|
content={
|
|
|
|
|
|
<div style={{ lineHeight: 1.6 }}>
|
|
|
|
|
|
<Typography.Text strong>{t('渠道亲和性')}</Typography.Text>
|
|
|
|
|
|
<div>
|
|
|
|
|
|
<Typography.Text type='secondary'>
|
|
|
|
|
|
{t('规则')}:{affinity.rule_name || '-'}
|
|
|
|
|
|
</Typography.Text>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
<div>
|
|
|
|
|
|
<Typography.Text type='secondary'>
|
|
|
|
|
|
{t('分组')}:{affinity.selected_group || '-'}
|
|
|
|
|
|
</Typography.Text>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
<div>
|
|
|
|
|
|
<Typography.Text type='secondary'>
|
|
|
|
|
|
{t('Key')}:
|
|
|
|
|
|
{(affinity.key_source || '-') +
|
|
|
|
|
|
':' +
|
|
|
|
|
|
(affinity.key_path || affinity.key_key || '-') +
|
|
|
|
|
|
(affinity.key_fp ? `#${affinity.key_fp}` : '')}
|
|
|
|
|
|
</Typography.Text>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
}
|
|
|
|
|
|
>
|
|
|
|
|
|
<span>
|
|
|
|
|
|
<Tag className='channel-affinity-tag' color='cyan' shape='circle'>
|
|
|
|
|
|
<span className='channel-affinity-tag-content'>
|
|
|
|
|
|
<IconStarStroked style={{ fontSize: 13 }} />
|
|
|
|
|
|
{t('优选')}
|
|
|
|
|
|
</span>
|
|
|
|
|
|
</Tag>
|
|
|
|
|
|
</span>
|
|
|
|
|
|
</Tooltip>
|
|
|
|
|
|
) : null}
|
|
|
|
|
|
</Space>
|
|
|
|
|
|
) : (
|
|
|
|
|
|
<></>
|
|
|
|
|
|
);
|
2025-07-18 22:04:54 +08:00
|
|
|
|
},
|
|
|
|
|
|
},
|
|
|
|
|
|
{
|
|
|
|
|
|
key: COLUMN_KEYS.DETAILS,
|
|
|
|
|
|
title: t('详情'),
|
|
|
|
|
|
dataIndex: 'content',
|
|
|
|
|
|
fixed: 'right',
|
|
|
|
|
|
render: (text, record, index) => {
|
|
|
|
|
|
let other = getLogOther(record.other);
|
|
|
|
|
|
if (other == null || record.type !== 2) {
|
|
|
|
|
|
return (
|
|
|
|
|
|
<Typography.Paragraph
|
|
|
|
|
|
ellipsis={{
|
|
|
|
|
|
rows: 2,
|
|
|
|
|
|
showTooltip: {
|
|
|
|
|
|
type: 'popover',
|
|
|
|
|
|
opts: { style: { width: 240 } },
|
|
|
|
|
|
},
|
|
|
|
|
|
}}
|
|
|
|
|
|
style={{ maxWidth: 240 }}
|
|
|
|
|
|
>
|
|
|
|
|
|
{text}
|
|
|
|
|
|
</Typography.Paragraph>
|
|
|
|
|
|
);
|
|
|
|
|
|
}
|
2026-01-26 20:20:30 +08:00
|
|
|
|
|
|
|
|
|
|
if (
|
|
|
|
|
|
other?.violation_fee === true ||
|
|
|
|
|
|
Boolean(other?.violation_fee_code) ||
|
|
|
|
|
|
Boolean(other?.violation_fee_marker)
|
|
|
|
|
|
) {
|
|
|
|
|
|
const feeQuota = other?.fee_quota ?? record?.quota;
|
|
|
|
|
|
const ratioText = formatRatio(other?.group_ratio);
|
|
|
|
|
|
const summary = [
|
|
|
|
|
|
t('违规扣费'),
|
|
|
|
|
|
`${t('分组倍率')}:${ratioText}`,
|
|
|
|
|
|
`${t('扣费')}:${renderQuota(feeQuota, 6)}`,
|
|
|
|
|
|
text ? `${t('详情')}:${text}` : null,
|
|
|
|
|
|
]
|
|
|
|
|
|
.filter(Boolean)
|
|
|
|
|
|
.join('\n');
|
|
|
|
|
|
return (
|
|
|
|
|
|
<Typography.Paragraph
|
|
|
|
|
|
ellipsis={{
|
|
|
|
|
|
rows: 2,
|
|
|
|
|
|
showTooltip: {
|
|
|
|
|
|
type: 'popover',
|
|
|
|
|
|
opts: { style: { width: 240 } },
|
|
|
|
|
|
},
|
|
|
|
|
|
}}
|
|
|
|
|
|
style={{ maxWidth: 240, whiteSpace: 'pre-line' }}
|
|
|
|
|
|
>
|
|
|
|
|
|
{summary}
|
|
|
|
|
|
</Typography.Paragraph>
|
|
|
|
|
|
);
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2025-07-18 22:04:54 +08:00
|
|
|
|
let content = other?.claude
|
2025-08-09 12:53:06 +08:00
|
|
|
|
? renderModelPriceSimple(
|
2025-08-30 21:15:10 +08:00
|
|
|
|
other.model_ratio,
|
|
|
|
|
|
other.model_price,
|
|
|
|
|
|
other.group_ratio,
|
|
|
|
|
|
other?.user_group_ratio,
|
|
|
|
|
|
other.cache_tokens || 0,
|
|
|
|
|
|
other.cache_ratio || 1.0,
|
|
|
|
|
|
other.cache_creation_tokens || 0,
|
|
|
|
|
|
other.cache_creation_ratio || 1.0,
|
2025-11-04 00:20:50 +08:00
|
|
|
|
other.cache_creation_tokens_5m || 0,
|
2026-01-26 19:57:41 +08:00
|
|
|
|
other.cache_creation_ratio_5m ||
|
|
|
|
|
|
other.cache_creation_ratio ||
|
|
|
|
|
|
1.0,
|
2025-11-04 00:20:50 +08:00
|
|
|
|
other.cache_creation_tokens_1h || 0,
|
2026-01-26 19:57:41 +08:00
|
|
|
|
other.cache_creation_ratio_1h ||
|
|
|
|
|
|
other.cache_creation_ratio ||
|
|
|
|
|
|
1.0,
|
2025-08-30 21:15:10 +08:00
|
|
|
|
false,
|
|
|
|
|
|
1.0,
|
|
|
|
|
|
other?.is_system_prompt_overwritten,
|
|
|
|
|
|
'claude',
|
|
|
|
|
|
)
|
2025-07-18 22:04:54 +08:00
|
|
|
|
: renderModelPriceSimple(
|
2025-08-30 21:15:10 +08:00
|
|
|
|
other.model_ratio,
|
|
|
|
|
|
other.model_price,
|
|
|
|
|
|
other.group_ratio,
|
|
|
|
|
|
other?.user_group_ratio,
|
|
|
|
|
|
other.cache_tokens || 0,
|
|
|
|
|
|
other.cache_ratio || 1.0,
|
|
|
|
|
|
0,
|
|
|
|
|
|
1.0,
|
2025-11-04 00:20:50 +08:00
|
|
|
|
0,
|
|
|
|
|
|
1.0,
|
|
|
|
|
|
0,
|
|
|
|
|
|
1.0,
|
2025-08-30 21:15:10 +08:00
|
|
|
|
false,
|
|
|
|
|
|
1.0,
|
|
|
|
|
|
other?.is_system_prompt_overwritten,
|
|
|
|
|
|
'openai',
|
|
|
|
|
|
);
|
2025-07-18 22:04:54 +08:00
|
|
|
|
return (
|
|
|
|
|
|
<Typography.Paragraph
|
|
|
|
|
|
ellipsis={{
|
2025-08-09 12:53:06 +08:00
|
|
|
|
rows: 3,
|
2025-07-18 22:04:54 +08:00
|
|
|
|
}}
|
2025-08-09 12:53:06 +08:00
|
|
|
|
style={{ maxWidth: 240, whiteSpace: 'pre-line' }}
|
2025-07-18 22:04:54 +08:00
|
|
|
|
>
|
|
|
|
|
|
{content}
|
|
|
|
|
|
</Typography.Paragraph>
|
|
|
|
|
|
);
|
|
|
|
|
|
},
|
|
|
|
|
|
},
|
|
|
|
|
|
];
|
2025-08-30 21:15:10 +08:00
|
|
|
|
};
|