mirror of
https://github.com/Wei-Shaw/sub2api.git
synced 2026-03-30 04:25:09 +00:00
- 新增管理端接口 /api/v1/admin/ops/dashboard/openai-token-stats,按模型聚合统计 gpt% 请求 - 支持 time_range=30m|1h|1d|15d|30d(默认 30d),支持 platform/group_id 过滤 - 支持分页(page/page_size)或 TopN(top_n)互斥查询 - 前端运维监控页新增统计表卡片,包含空态/错误态与分页/TopN 交互 - 补齐后端与前端测试
247 lines
8.5 KiB
Vue
247 lines
8.5 KiB
Vue
<script setup lang="ts">
|
||
import { computed, ref, watch } from 'vue'
|
||
import { useI18n } from 'vue-i18n'
|
||
import Select from '@/components/common/Select.vue'
|
||
import EmptyState from '@/components/common/EmptyState.vue'
|
||
import { opsAPI, type OpsOpenAITokenStatsResponse, type OpsOpenAITokenStatsTimeRange } from '@/api/admin/ops'
|
||
import { formatNumber } from '@/utils/format'
|
||
|
||
interface Props {
|
||
platformFilter?: string
|
||
groupIdFilter?: number | null
|
||
refreshToken: number
|
||
}
|
||
|
||
type ViewMode = 'topn' | 'pagination'
|
||
|
||
const props = withDefaults(defineProps<Props>(), {
|
||
platformFilter: '',
|
||
groupIdFilter: null
|
||
})
|
||
|
||
const { t } = useI18n()
|
||
|
||
const loading = ref(false)
|
||
const errorMessage = ref('')
|
||
const response = ref<OpsOpenAITokenStatsResponse | null>(null)
|
||
|
||
const timeRange = ref<OpsOpenAITokenStatsTimeRange>('30d')
|
||
const viewMode = ref<ViewMode>('topn')
|
||
const topN = ref<number>(20)
|
||
const page = ref<number>(1)
|
||
const pageSize = ref<number>(20)
|
||
|
||
const items = computed(() => response.value?.items ?? [])
|
||
const total = computed(() => response.value?.total ?? 0)
|
||
const totalPages = computed(() => {
|
||
if (viewMode.value !== 'pagination') return 1
|
||
const size = pageSize.value > 0 ? pageSize.value : 20
|
||
return Math.max(1, Math.ceil(total.value / size))
|
||
})
|
||
|
||
const timeRangeOptions = computed(() => [
|
||
{ value: '30m', label: t('admin.ops.timeRange.30m') },
|
||
{ value: '1h', label: t('admin.ops.timeRange.1h') },
|
||
{ value: '1d', label: t('admin.ops.timeRange.1d') },
|
||
{ value: '15d', label: t('admin.ops.timeRange.15d') },
|
||
{ value: '30d', label: t('admin.ops.timeRange.30d') }
|
||
])
|
||
|
||
const viewModeOptions = computed(() => [
|
||
{ value: 'topn', label: t('admin.ops.openaiTokenStats.viewModeTopN') },
|
||
{ value: 'pagination', label: t('admin.ops.openaiTokenStats.viewModePagination') }
|
||
])
|
||
|
||
const topNOptions = computed(() => [
|
||
{ value: 10, label: 'Top 10' },
|
||
{ value: 20, label: 'Top 20' },
|
||
{ value: 50, label: 'Top 50' },
|
||
{ value: 100, label: 'Top 100' }
|
||
])
|
||
|
||
const pageSizeOptions = computed(() => [
|
||
{ value: 10, label: '10' },
|
||
{ value: 20, label: '20' },
|
||
{ value: 50, label: '50' },
|
||
{ value: 100, label: '100' }
|
||
])
|
||
|
||
function formatRate(v?: number | null): string {
|
||
if (typeof v !== 'number' || !Number.isFinite(v)) return '-'
|
||
return v.toFixed(2)
|
||
}
|
||
|
||
function formatInt(v?: number | null): string {
|
||
if (typeof v !== 'number' || !Number.isFinite(v)) return '-'
|
||
return formatNumber(Math.round(v))
|
||
}
|
||
|
||
function buildParams() {
|
||
const params: Record<string, any> = {
|
||
time_range: timeRange.value,
|
||
platform: props.platformFilter || undefined,
|
||
group_id: typeof props.groupIdFilter === 'number' && props.groupIdFilter > 0 ? props.groupIdFilter : undefined
|
||
}
|
||
|
||
if (viewMode.value === 'topn') {
|
||
params.top_n = topN.value
|
||
} else {
|
||
params.page = page.value
|
||
params.page_size = pageSize.value
|
||
}
|
||
return params
|
||
}
|
||
|
||
async function loadData() {
|
||
loading.value = true
|
||
errorMessage.value = ''
|
||
try {
|
||
response.value = await opsAPI.getOpenAITokenStats(buildParams())
|
||
// 防御:若 total 变化导致当前页超出最大页,则回退到末页并重新拉取一次。
|
||
if (viewMode.value === 'pagination' && page.value > totalPages.value) {
|
||
page.value = totalPages.value
|
||
response.value = await opsAPI.getOpenAITokenStats(buildParams())
|
||
}
|
||
} catch (err: any) {
|
||
console.error('[OpsOpenAITokenStatsCard] Failed to load data', err)
|
||
response.value = null
|
||
errorMessage.value = err?.message || t('admin.ops.openaiTokenStats.failedToLoad')
|
||
} finally {
|
||
loading.value = false
|
||
}
|
||
}
|
||
|
||
watch(
|
||
() => ({
|
||
timeRange: timeRange.value,
|
||
viewMode: viewMode.value,
|
||
topN: topN.value,
|
||
page: page.value,
|
||
pageSize: pageSize.value,
|
||
platform: props.platformFilter,
|
||
groupId: props.groupIdFilter,
|
||
refreshToken: props.refreshToken
|
||
}),
|
||
(next, prev) => {
|
||
// 避免“筛选变化 -> 重置页码 -> 触发两次请求”:
|
||
// 先只重置页码,等待下一次 watch(仅 page 变化)再发起请求。
|
||
const filtersChanged = !prev ||
|
||
next.timeRange !== prev.timeRange ||
|
||
next.viewMode !== prev.viewMode ||
|
||
next.pageSize !== prev.pageSize ||
|
||
next.platform !== prev.platform ||
|
||
next.groupId !== prev.groupId ||
|
||
next.refreshToken !== prev.refreshToken
|
||
|
||
if (next.viewMode === 'pagination' && filtersChanged && next.page !== 1) {
|
||
page.value = 1
|
||
return
|
||
}
|
||
|
||
void loadData()
|
||
},
|
||
{ immediate: true }
|
||
)
|
||
|
||
function onPrevPage() {
|
||
if (viewMode.value !== 'pagination') return
|
||
if (page.value > 1) page.value -= 1
|
||
}
|
||
|
||
function onNextPage() {
|
||
if (viewMode.value !== 'pagination') return
|
||
if (page.value < totalPages.value) page.value += 1
|
||
}
|
||
</script>
|
||
|
||
<template>
|
||
<section class="card p-4 md:p-5">
|
||
<div class="mb-4 flex flex-wrap items-center justify-between gap-3">
|
||
<h3 class="text-sm font-bold text-gray-900 dark:text-white">
|
||
{{ t('admin.ops.openaiTokenStats.title') }}
|
||
</h3>
|
||
<div class="flex flex-wrap items-center gap-2">
|
||
<div class="w-36">
|
||
<Select v-model="timeRange" :options="timeRangeOptions" />
|
||
</div>
|
||
<div class="w-36">
|
||
<Select v-model="viewMode" :options="viewModeOptions" />
|
||
</div>
|
||
<div v-if="viewMode === 'topn'" class="w-28">
|
||
<Select v-model="topN" :options="topNOptions" />
|
||
</div>
|
||
<template v-else>
|
||
<div class="w-24">
|
||
<Select v-model="pageSize" :options="pageSizeOptions" />
|
||
</div>
|
||
<button
|
||
class="btn btn-secondary btn-sm"
|
||
:disabled="loading || page <= 1"
|
||
@click="onPrevPage"
|
||
>
|
||
{{ t('admin.ops.openaiTokenStats.prevPage') }}
|
||
</button>
|
||
<button
|
||
class="btn btn-secondary btn-sm"
|
||
:disabled="loading || page >= totalPages"
|
||
@click="onNextPage"
|
||
>
|
||
{{ t('admin.ops.openaiTokenStats.nextPage') }}
|
||
</button>
|
||
<span class="text-xs text-gray-500 dark:text-gray-400">
|
||
{{ t('admin.ops.openaiTokenStats.pageInfo', { page, total: totalPages }) }}
|
||
</span>
|
||
</template>
|
||
</div>
|
||
</div>
|
||
|
||
<div v-if="errorMessage" class="mb-4 rounded-lg bg-red-50 px-3 py-2 text-xs text-red-600 dark:bg-red-900/20 dark:text-red-400">
|
||
{{ errorMessage }}
|
||
</div>
|
||
|
||
<div v-if="loading" class="py-8 text-center text-sm text-gray-500 dark:text-gray-400">
|
||
{{ t('admin.ops.loadingText') }}
|
||
</div>
|
||
|
||
<EmptyState
|
||
v-else-if="items.length === 0"
|
||
:title="t('common.noData')"
|
||
:description="t('admin.ops.openaiTokenStats.empty')"
|
||
/>
|
||
|
||
<div v-else class="overflow-x-auto">
|
||
<table class="min-w-full text-left text-xs md:text-sm">
|
||
<thead>
|
||
<tr class="border-b border-gray-200 text-gray-500 dark:border-dark-700 dark:text-gray-400">
|
||
<th class="px-2 py-2 font-semibold">{{ t('admin.ops.openaiTokenStats.table.model') }}</th>
|
||
<th class="px-2 py-2 font-semibold">{{ t('admin.ops.openaiTokenStats.table.requestCount') }}</th>
|
||
<th class="px-2 py-2 font-semibold">{{ t('admin.ops.openaiTokenStats.table.avgTokensPerSec') }}</th>
|
||
<th class="px-2 py-2 font-semibold">{{ t('admin.ops.openaiTokenStats.table.avgFirstTokenMs') }}</th>
|
||
<th class="px-2 py-2 font-semibold">{{ t('admin.ops.openaiTokenStats.table.totalOutputTokens') }}</th>
|
||
<th class="px-2 py-2 font-semibold">{{ t('admin.ops.openaiTokenStats.table.avgDurationMs') }}</th>
|
||
<th class="px-2 py-2 font-semibold">{{ t('admin.ops.openaiTokenStats.table.requestsWithFirstToken') }}</th>
|
||
</tr>
|
||
</thead>
|
||
<tbody>
|
||
<tr
|
||
v-for="row in items"
|
||
:key="row.model"
|
||
class="border-b border-gray-100 text-gray-700 dark:border-dark-800 dark:text-gray-200"
|
||
>
|
||
<td class="px-2 py-2 font-medium">{{ row.model }}</td>
|
||
<td class="px-2 py-2">{{ formatInt(row.request_count) }}</td>
|
||
<td class="px-2 py-2">{{ formatRate(row.avg_tokens_per_sec) }}</td>
|
||
<td class="px-2 py-2">{{ formatRate(row.avg_first_token_ms) }}</td>
|
||
<td class="px-2 py-2">{{ formatInt(row.total_output_tokens) }}</td>
|
||
<td class="px-2 py-2">{{ formatInt(row.avg_duration_ms) }}</td>
|
||
<td class="px-2 py-2">{{ formatInt(row.requests_with_first_token) }}</td>
|
||
</tr>
|
||
</tbody>
|
||
</table>
|
||
<div v-if="viewMode === 'topn'" class="mt-3 text-xs text-gray-500 dark:text-gray-400">
|
||
{{ t('admin.ops.openaiTokenStats.totalModels', { total }) }}
|
||
</div>
|
||
</div>
|
||
</section>
|
||
</template>
|