| <template> |
| <div class="benchmark-app"> |
| |
| <div class="main-container"> |
|
|
| |
| <header class="header"> |
| <a href="https://docs.sglang.io/SpecForge/" class="back-to-docs"> |
| <svg width="18" height="18" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round"> |
| <path d="M19 12H5M12 19l-7-7 7-7"/> |
| </svg> |
| <span>Back to SpecForge Docs</span> |
| </a> |
| <div class="header-content"> |
| <div class="brand"> |
| <img src="/specbundle-logo.png" alt="SpecBundle Logo" class="main-logo"> |
| <div class="powered-by"> |
| All SpecBundle models are powered by <span class="powered-by-brand">SpecForge</span> |
| </div> |
| <a href="https://huggingface.co/collections/lmsys/specbundle" target="_blank" rel="noopener noreferrer" class="models-link"> |
| <svg width="16" height="16" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round"> |
| <path d="M18 13v6a2 2 0 0 1-2 2H5a2 2 0 0 1-2-2V8a2 2 0 0 1 2-2h6"></path> |
| <polyline points="15 3 21 3 21 9"></polyline> |
| <line x1="10" y1="14" x2="21" y2="3"></line> |
| </svg> |
| View All Models on Hugging Face |
| </a> |
| </div> |
| </div> |
| </header> |
|
|
| |
| <div class="dashboard-controls"> |
| <FilterControls |
| :targetModels="targetModels" |
| :selectedTargetModel="selectedTargetModel" |
| :draftModels="draftModels" |
| :selectedDraftModel="selectedDraftModel" |
| :benchmarks="benchmarks" |
| :selectedBenchmark="selectedBenchmark" |
| :metrics="metricOptions" |
| :selectedMetric="selectedMetric" |
| @update:targetModel="selectedTargetModel = $event" |
| @update:draftModel="selectedDraftModel = $event" |
| @update:benchmark="selectedBenchmark = $event" |
| @update:metric="selectedMetric = $event" |
| /> |
| </div> |
|
|
| |
| |
|
|
| |
| <div v-if="loading" class="state-container"> |
| <div class="spinner"></div> |
| <p>Loading benchmark data...</p> |
| </div> |
|
|
| |
| <div v-else-if="currentData.length > 0" class="content-wrapper"> |
| <div class="stats-overview"> |
| <div class="stat-card"> |
| <span class="stat-label">Target Model</span> |
| <span class="stat-value">{{ removeSGLangPrefix(selectedTargetModel) }}</span> |
| </div> |
| <div class="stat-card"> |
| <span class="stat-label">Draft Models</span> |
| <span class="stat-value">{{ draftModels.length }}</span> |
| </div> |
| <div class="stat-card"> |
| <span class="stat-label">Configurations</span> |
| <span class="stat-value">{{ currentData.length }}</span> |
| </div> |
| </div> |
|
|
| <div class="chart-section"> |
| <h3>Performance Visualization</h3> |
| <div class="chart-container"> |
| <BenchmarkChart |
| :data="currentData" |
| :benchmark="selectedBenchmark" |
| :metric="selectedMetric" |
| /> |
| </div> |
| <div class="config-legend">Config (batch_size - step - topk - nums_draft_tokens)</div> |
| </div> |
|
|
| <div class="table-section"> |
| <div class="section-header"> |
| <h3>Detailed Results</h3> |
| <div class="export-actions"> |
| |
| </div> |
| </div> |
| <BenchmarkTable |
| :data="currentData" |
| :benchmarks="benchmarks" |
| :selectedBenchmark="selectedBenchmark" |
| :highlightMetric="selectedMetric" |
| /> |
| </div> |
| </div> |
|
|
|
|
|
|
| |
| <div v-else class="state-container empty"> |
| <div class="empty-icon"> |
| <svg width="48" height="48" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="1.5"> |
| <path d="M14 2H6a2 2 0 0 0-2 2v16a2 2 0 0 0 2 2h12a2 2 0 0 0 2-2V8z"></path> |
| <polyline points="14 2 14 8 20 8"></polyline> |
| <line x1="16" y1="13" x2="8" y2="13"></line> |
| <line x1="16" y1="17" x2="8" y2="17"></line> |
| <polyline points="10 9 9 9 8 9"></polyline> |
| </svg> |
| </div> |
| <h3>No Data Available</h3> |
| <p>No benchmark data found for {{ selectedTargetModel }}</p> |
| </div> |
|
|
| </div> |
|
|
| <footer class="footer"> |
| <p>Powered by SpecForge Team</p> |
| </footer> |
| </div> |
| </template> |
|
|
| <script setup> |
| import { ref, computed, onMounted, watch } from 'vue'; |
| import FilterControls from './FilterControls.vue'; |
| import BenchmarkChart from './BenchmarkChart.vue'; |
| import BenchmarkTable from './BenchmarkTable.vue'; |
| import { |
| loadAllData, |
| processModelData, |
| getTargetModels, |
| removeSGLangPrefix |
| } from '../utils/dataProcessor'; |
| |
| const loading = ref(true); |
| const allProcessedData = ref({}); |
| const selectedTargetModel = ref(''); |
| const selectedDraftModel = ref('all'); |
| const selectedBenchmark = ref('all'); |
| const selectedMetric = ref('throughput'); |
| |
| |
| const benchmarks = ['gsm8k', 'math500', 'mtbench', 'humaneval', 'livecodebench', 'financeqa', 'gpqa']; |
| const metricOptions = [ |
| { value: 'throughput', label: 'Throughput (tokens/s)' }, |
| { value: 'accLen', label: 'Acceptance Length' }, |
| { value: 'speedup', label: 'Speedup vs Baseline' } |
| ]; |
| |
| const targetModels = computed(() => getTargetModels(allProcessedData.value)); |
| |
| const draftModels = computed(() => { |
| if (!selectedTargetModel.value) return []; |
| const targetData = allProcessedData.value[selectedTargetModel.value] || []; |
| return [...new Set(targetData.map(d => d.draftModel).filter(Boolean))]; |
| }); |
| |
| const currentData = computed(() => { |
| const targetData = allProcessedData.value[selectedTargetModel.value] || []; |
| |
| if (selectedDraftModel.value && selectedDraftModel.value !== 'all') { |
| return targetData.filter(d => d.draftModel === selectedDraftModel.value); |
| } |
| |
| return targetData; |
| }); |
| |
| |
| watch(selectedTargetModel, (newTargetModel) => { |
| if (newTargetModel) { |
| const newDraftModels = draftModels.value; |
| if (!newDraftModels.includes(selectedDraftModel.value)) { |
| selectedDraftModel.value = 'all'; |
| } |
| } |
| }); |
| |
| onMounted(async () => { |
| try { |
| const jsonData = await loadAllData(); |
| |
| |
| for (const [targetModel, modelData] of Object.entries(jsonData)) { |
| allProcessedData.value[targetModel] = processModelData(modelData, targetModel); |
| } |
| |
| |
| const models = Object.keys(allProcessedData.value); |
| if (models.length > 0) { |
| selectedTargetModel.value = models[0]; |
| } |
| |
| loading.value = false; |
| } catch (error) { |
| console.error('Error loading data:', error); |
| loading.value = false; |
| } |
| }); |
| </script> |
|
|
| <style scoped> |
| .benchmark-app { |
| min-height: 100vh; |
| display: flex; |
| flex-direction: column; |
| background: #F8FAFC; |
| } |
| |
| .main-container { |
| max-width: 1440px; |
| margin: 0 auto; |
| padding: 40px 24px; |
| width: 100%; |
| flex: 1; |
| } |
| |
| |
| .header { |
| margin-bottom: 40px; |
| text-align: center; |
| position: relative; |
| } |
| |
| .back-to-docs { |
| position: absolute; |
| top: 0; |
| left: 0; |
| display: inline-flex; |
| align-items: center; |
| gap: 8px; |
| padding: 10px 16px; |
| background: white; |
| color: var(--color-text-secondary); |
| text-decoration: none; |
| border-radius: 8px; |
| font-size: 0.875rem; |
| font-weight: 500; |
| transition: all 0.3s cubic-bezier(0.4, 0, 0.2, 1); |
| box-shadow: 0 2px 8px rgba(0, 0, 0, 0.08); |
| border: 1px solid #e2e8f0; |
| } |
| |
| .back-to-docs:hover { |
| color: var(--color-primary); |
| border-color: var(--color-primary); |
| box-shadow: 0 4px 12px rgba(79, 70, 229, 0.15); |
| transform: translateX(-2px); |
| } |
| |
| .back-to-docs:active { |
| transform: translateX(0); |
| } |
| |
| .back-to-docs svg { |
| width: 18px; |
| height: 18px; |
| flex-shrink: 0; |
| transition: transform 0.3s ease; |
| } |
| |
| .back-to-docs:hover svg { |
| transform: translateX(-2px); |
| } |
| |
| .header-content { |
| display: flex; |
| flex-direction: column; |
| align-items: center; |
| justify-content: center; |
| gap: 1rem; |
| } |
| |
| .brand { |
| display: flex; |
| flex-direction: column; |
| align-items: center; |
| justify-content: center; |
| gap: 16px; |
| } |
| |
| .main-logo { |
| max-width: 700px; |
| width: auto; |
| height: 140px; |
| object-fit: contain; |
| } |
| |
| .powered-by { |
| font-size: 0.9375rem; |
| color: var(--color-text-secondary); |
| font-weight: 400; |
| text-align: center; |
| } |
| |
| .powered-by-brand { |
| color: var(--color-primary); |
| font-weight: 600; |
| } |
| |
| .models-link { |
| display: inline-flex; |
| align-items: center; |
| gap: 8px; |
| margin-top: 8px; |
| padding: 10px 20px; |
| background: var(--color-primary); |
| color: white; |
| text-decoration: none; |
| border-radius: 8px; |
| font-size: 0.875rem; |
| font-weight: 500; |
| transition: all 0.3s cubic-bezier(0.4, 0, 0.2, 1); |
| box-shadow: 0 2px 8px rgba(79, 70, 229, 0.2); |
| } |
| |
| .models-link:hover { |
| background: var(--color-primary-dark); |
| transform: translateY(-2px); |
| box-shadow: 0 4px 12px rgba(79, 70, 229, 0.3); |
| } |
| |
| .models-link:active { |
| transform: translateY(0); |
| } |
| |
| .models-link svg { |
| width: 16px; |
| height: 16px; |
| flex-shrink: 0; |
| } |
| |
| .config-legend { |
| font-size: 0.875rem; |
| color: var(--color-text-secondary); |
| font-weight: 500; |
| margin-top: 8px; |
| } |
| |
| .subtitle { |
| font-size: 1.25rem; |
| color: var(--color-text-secondary); |
| max-width: 600px; |
| text-align: center; |
| } |
| |
| |
| .dashboard-controls { |
| margin-bottom: 32px; |
| } |
| |
| |
| .tabs-container { |
| margin-bottom: 40px; |
| display: flex; |
| justify-content: center; |
| } |
| |
| .tabs-scroll { |
| display: flex; |
| gap: 8px; |
| padding: 8px; |
| background: white; |
| border-radius: var(--radius-xl); |
| box-shadow: var(--shadow-sm); |
| overflow-x: auto; |
| max-width: 100%; |
| border: 1px solid #e2e8f0; |
| } |
| |
| .tab-btn { |
| padding: 10px 20px; |
| border: none; |
| background: transparent; |
| color: var(--color-text-secondary); |
| font-weight: 600; |
| font-size: 0.9375rem; |
| border-radius: var(--radius-lg); |
| transition: all 0.2s ease; |
| white-space: nowrap; |
| } |
| |
| .tab-btn:hover { |
| color: var(--color-primary); |
| background: #f1f5f9; |
| } |
| |
| .tab-btn.active { |
| background: var(--color-text-main); |
| color: white; |
| box-shadow: 0 4px 12px rgba(15, 23, 42, 0.2); |
| } |
| |
| .tab-divider { |
| width: 1px; |
| background: #e2e8f0; |
| margin: 4px 8px; |
| } |
| |
| .tab-btn.comparison { |
| display: flex; |
| align-items: center; |
| gap: 8px; |
| color: var(--color-primary); |
| } |
| |
| .tab-btn.comparison.active { |
| background: var(--color-primary); |
| color: white; |
| box-shadow: 0 4px 12px rgba(79, 70, 229, 0.3); |
| } |
| |
| |
| .content-wrapper { |
| animation: fadeIn 0.5s cubic-bezier(0.16, 1, 0.3, 1); |
| } |
| |
| @keyframes fadeIn { |
| from { opacity: 0; transform: translateY(10px); } |
| to { opacity: 1; transform: translateY(0); } |
| } |
| |
| |
| .stats-overview { |
| display: grid; |
| grid-template-columns: repeat(auto-fit, minmax(200px, 1fr)); |
| gap: 24px; |
| margin-bottom: 32px; |
| } |
| |
| .stat-card { |
| background: white; |
| padding: 24px; |
| border-radius: var(--radius-xl); |
| box-shadow: var(--shadow-sm), inset 0 1px 0 0 rgba(255, 255, 255, 0.6); |
| display: flex; |
| flex-direction: column; |
| align-items: flex-start; |
| border: 1px solid rgba(255, 255, 255, 0.5); |
| transition: transform 0.3s cubic-bezier(0.4, 0, 0.2, 1), box-shadow 0.3s cubic-bezier(0.4, 0, 0.2, 1); |
| position: relative; |
| overflow: hidden; |
| } |
| |
| |
| |
| .stat-card:hover { |
| transform: translateY(-4px); |
| box-shadow: var(--shadow-xl); |
| } |
| |
| |
| .stat-card:not(.highlight):hover .stat-value, |
| .stat-card:not(.highlight):hover .stat-label { |
| color: var(--color-primary); |
| } |
| |
| .stat-card.highlight { |
| background: linear-gradient(135deg, var(--color-primary) 0%, var(--color-primary-dark) 100%); |
| box-shadow: var(--shadow-glow); |
| border: none; |
| } |
| |
| .stat-card.highlight .stat-label { |
| color: rgba(255,255,255, 0.8); |
| } |
| |
| .stat-card.highlight .stat-value { |
| color: white; |
| } |
| |
| .stat-label { |
| font-size: 0.75rem; |
| text-transform: uppercase; |
| letter-spacing: 0.1em; |
| color: var(--color-text-muted); |
| font-weight: 600; |
| margin-bottom: 8px; |
| transition: color 0.3s ease; |
| } |
| |
| .stat-value { |
| font-family: var(--font-display); |
| font-size: 2.25rem; |
| font-weight: 700; |
| color: var(--color-text-main); |
| line-height: 1; |
| letter-spacing: -0.02em; |
| transition: color 0.3s ease; |
| } |
| |
| |
| .chart-section, |
| .table-section { |
| background: white; |
| border-radius: var(--radius-2xl); |
| box-shadow: var(--shadow-lg), inset 0 1px 0 0 rgba(255, 255, 255, 0.6); |
| padding: 40px; |
| margin-bottom: 32px; |
| border: 1px solid rgba(255, 255, 255, 0.5); |
| } |
| |
| .chart-section h3, |
| .section-header h3 { |
| font-family: var(--font-display); |
| font-size: 1.5rem; |
| font-weight: 700; |
| color: var(--color-text-main); |
| margin-bottom: 8px; |
| } |
| |
| .chart-section .config-legend { |
| font-size: 0.875rem; |
| color: var(--color-text-secondary); |
| font-weight: 500; |
| margin-top: 16px; |
| text-align: center; |
| } |
| |
| .chart-description { |
| font-size: 0.95rem; |
| color: var(--color-text-secondary); |
| margin-bottom: 24px; |
| padding: 8px 16px; |
| background: linear-gradient(135deg, #f8fafc 0%, #f1f5f9 100%); |
| border-left: 3px solid var(--color-primary); |
| border-radius: 4px; |
| } |
| |
| .chart-container { |
| width: 100%; |
| height: 500px; |
| } |
| |
| |
| .state-container { |
| display: flex; |
| flex-direction: column; |
| align-items: center; |
| justify-content: center; |
| padding: 80px 0; |
| text-align: center; |
| } |
| |
| .spinner { |
| width: 50px; |
| height: 50px; |
| border: 4px solid #e2e8f0; |
| border-top-color: var(--color-primary); |
| border-radius: 50%; |
| animation: spin 1s linear infinite; |
| margin-bottom: 24px; |
| } |
| |
| @keyframes spin { |
| to { transform: rotate(360deg); } |
| } |
| |
| .empty-icon { |
| margin-bottom: 24px; |
| color: var(--color-text-muted); |
| } |
| |
| |
| .footer { |
| margin-top: auto; |
| padding: 40px 0; |
| text-align: center; |
| color: var(--color-text-muted); |
| font-size: 0.875rem; |
| border-top: 1px solid #e2e8f0; |
| } |
| |
| |
| @media (max-width: 768px) { |
| h1 { font-size: 2rem; } |
| .main-container { padding: 20px 16px; } |
| .chart-section, .table-section { padding: 20px; } |
| .tabs-scroll { flex-wrap: wrap; } |
| .back-to-docs { |
| position: relative; |
| top: auto; |
| left: auto; |
| margin-bottom: 20px; |
| align-self: flex-start; |
| } |
| .back-to-docs span { |
| display: none; |
| } |
| .header { |
| display: flex; |
| flex-direction: column; |
| align-items: center; |
| } |
| } |
| </style> |
|
|