Add a doughnut chart showing usage statistics broken down by group on the admin usage records page. The chart appears alongside the existing model distribution chart (2-column grid), with the token usage trend chart moved to a separate full-width row below. Changes: - backend/pkg/usagestats: add GroupStat type - backend/service: add GetGroupStatsWithFilters interface method and implementation - backend/repository: implement GetGroupStatsWithFilters with LEFT JOIN groups - backend/handler: add GetGroupStats handler with full filter support - backend/routes: register GET /admin/dashboard/groups route - backend/tests: add GetGroupStatsWithFilters stubs to contract/sora tests - frontend/types: add GroupStat interface - frontend/api: add getGroupStats API function and types - frontend/components: add GroupDistributionChart.vue doughnut chart - frontend/views: update UsageView layout and load group stats in parallel - frontend/i18n: add groupDistribution, group, noGroup keys (zh + en)
153 lines
4.5 KiB
Vue
153 lines
4.5 KiB
Vue
<template>
|
|
<div class="card p-4">
|
|
<h3 class="mb-4 text-sm font-semibold text-gray-900 dark:text-white">
|
|
{{ t('admin.dashboard.groupDistribution') }}
|
|
</h3>
|
|
<div v-if="loading" class="flex h-48 items-center justify-center">
|
|
<LoadingSpinner />
|
|
</div>
|
|
<div v-else-if="groupStats.length > 0 && chartData" class="flex items-center gap-6">
|
|
<div class="h-48 w-48">
|
|
<Doughnut :data="chartData" :options="doughnutOptions" />
|
|
</div>
|
|
<div class="max-h-48 flex-1 overflow-y-auto">
|
|
<table class="w-full text-xs">
|
|
<thead>
|
|
<tr class="text-gray-500 dark:text-gray-400">
|
|
<th class="pb-2 text-left">{{ t('admin.dashboard.group') }}</th>
|
|
<th class="pb-2 text-right">{{ t('admin.dashboard.requests') }}</th>
|
|
<th class="pb-2 text-right">{{ t('admin.dashboard.tokens') }}</th>
|
|
<th class="pb-2 text-right">{{ t('admin.dashboard.actual') }}</th>
|
|
<th class="pb-2 text-right">{{ t('admin.dashboard.standard') }}</th>
|
|
</tr>
|
|
</thead>
|
|
<tbody>
|
|
<tr
|
|
v-for="group in groupStats"
|
|
:key="group.group_id"
|
|
class="border-t border-gray-100 dark:border-gray-700"
|
|
>
|
|
<td
|
|
class="max-w-[100px] truncate py-1.5 font-medium text-gray-900 dark:text-white"
|
|
:title="group.group_name || String(group.group_id)"
|
|
>
|
|
{{ group.group_name || t('admin.dashboard.noGroup') }}
|
|
</td>
|
|
<td class="py-1.5 text-right text-gray-600 dark:text-gray-400">
|
|
{{ formatNumber(group.requests) }}
|
|
</td>
|
|
<td class="py-1.5 text-right text-gray-600 dark:text-gray-400">
|
|
{{ formatTokens(group.total_tokens) }}
|
|
</td>
|
|
<td class="py-1.5 text-right text-green-600 dark:text-green-400">
|
|
${{ formatCost(group.actual_cost) }}
|
|
</td>
|
|
<td class="py-1.5 text-right text-gray-400 dark:text-gray-500">
|
|
${{ formatCost(group.cost) }}
|
|
</td>
|
|
</tr>
|
|
</tbody>
|
|
</table>
|
|
</div>
|
|
</div>
|
|
<div
|
|
v-else
|
|
class="flex h-48 items-center justify-center text-sm text-gray-500 dark:text-gray-400"
|
|
>
|
|
{{ t('admin.dashboard.noDataAvailable') }}
|
|
</div>
|
|
</div>
|
|
</template>
|
|
|
|
<script setup lang="ts">
|
|
import { computed } from 'vue'
|
|
import { useI18n } from 'vue-i18n'
|
|
import { Chart as ChartJS, ArcElement, Tooltip, Legend } from 'chart.js'
|
|
import { Doughnut } from 'vue-chartjs'
|
|
import LoadingSpinner from '@/components/common/LoadingSpinner.vue'
|
|
import type { GroupStat } from '@/types'
|
|
|
|
ChartJS.register(ArcElement, Tooltip, Legend)
|
|
|
|
const { t } = useI18n()
|
|
|
|
const props = defineProps<{
|
|
groupStats: GroupStat[]
|
|
loading?: boolean
|
|
}>()
|
|
|
|
const chartColors = [
|
|
'#3b82f6',
|
|
'#10b981',
|
|
'#f59e0b',
|
|
'#ef4444',
|
|
'#8b5cf6',
|
|
'#ec4899',
|
|
'#14b8a6',
|
|
'#f97316',
|
|
'#6366f1',
|
|
'#84cc16'
|
|
]
|
|
|
|
const chartData = computed(() => {
|
|
if (!props.groupStats?.length) return null
|
|
|
|
return {
|
|
labels: props.groupStats.map((g) => g.group_name || String(g.group_id)),
|
|
datasets: [
|
|
{
|
|
data: props.groupStats.map((g) => g.total_tokens),
|
|
backgroundColor: chartColors.slice(0, props.groupStats.length),
|
|
borderWidth: 0
|
|
}
|
|
]
|
|
}
|
|
})
|
|
|
|
const doughnutOptions = computed(() => ({
|
|
responsive: true,
|
|
maintainAspectRatio: false,
|
|
plugins: {
|
|
legend: {
|
|
display: false
|
|
},
|
|
tooltip: {
|
|
callbacks: {
|
|
label: (context: any) => {
|
|
const value = context.raw as number
|
|
const total = context.dataset.data.reduce((a: number, b: number) => a + b, 0)
|
|
const percentage = ((value / total) * 100).toFixed(1)
|
|
return `${context.label}: ${formatTokens(value)} (${percentage}%)`
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}))
|
|
|
|
const formatTokens = (value: number): string => {
|
|
if (value >= 1_000_000_000) {
|
|
return `${(value / 1_000_000_000).toFixed(2)}B`
|
|
} else if (value >= 1_000_000) {
|
|
return `${(value / 1_000_000).toFixed(2)}M`
|
|
} else if (value >= 1_000) {
|
|
return `${(value / 1_000).toFixed(2)}K`
|
|
}
|
|
return value.toLocaleString()
|
|
}
|
|
|
|
const formatNumber = (value: number): string => {
|
|
return value.toLocaleString()
|
|
}
|
|
|
|
const formatCost = (value: number): string => {
|
|
if (value >= 1000) {
|
|
return (value / 1000).toFixed(2) + 'K'
|
|
} else if (value >= 1) {
|
|
return value.toFixed(2)
|
|
} else if (value >= 0.01) {
|
|
return value.toFixed(3)
|
|
}
|
|
return value.toFixed(4)
|
|
}
|
|
</script>
|