229 lines
12 KiB
XML
229 lines
12 KiB
XML
<?xml version="1.0" encoding="UTF-8"?>
|
|
<templates xml:space="preserve">
|
|
<t t-name="fusion_api.Dashboard">
|
|
<div class="o_fusion_api_dashboard o_action">
|
|
<!-- Header -->
|
|
<div class="o_fusion_api_header d-flex align-items-center justify-content-between p-3 border-bottom bg-view">
|
|
<h2 class="mb-0">Fusion API Dashboard</h2>
|
|
<button class="btn btn-outline-primary" t-on-click="refresh">
|
|
<i class="fa fa-refresh me-1"/> Refresh
|
|
</button>
|
|
</div>
|
|
|
|
<div class="o_fusion_api_content p-3" t-if="state.loaded">
|
|
<!-- Stats Cards -->
|
|
<div class="row g-3 mb-4">
|
|
<div class="col-md-3 col-sm-6">
|
|
<div class="card h-100 border-0 shadow-sm o_fusion_stat_card"
|
|
style="cursor:pointer" t-on-click="openProviders">
|
|
<div class="card-body">
|
|
<div class="d-flex justify-content-between align-items-start">
|
|
<div>
|
|
<div class="text-muted small text-uppercase">Active Providers</div>
|
|
<h2 class="mb-0 mt-1" t-esc="state.activeProviders"/>
|
|
</div>
|
|
<div class="o_fusion_icon_circle bg-primary bg-opacity-25">
|
|
<i class="fa fa-plug text-primary"/>
|
|
</div>
|
|
</div>
|
|
<div class="text-muted small mt-2">
|
|
<t t-esc="state.totalProviders"/> total configured
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
|
|
<div class="col-md-3 col-sm-6">
|
|
<div class="card h-100 border-0 shadow-sm o_fusion_stat_card"
|
|
style="cursor:pointer" t-on-click="openConsumers">
|
|
<div class="card-body">
|
|
<div class="d-flex justify-content-between align-items-start">
|
|
<div>
|
|
<div class="text-muted small text-uppercase">Active Modules</div>
|
|
<h2 class="mb-0 mt-1" t-esc="state.activeConsumers"/>
|
|
</div>
|
|
<div class="o_fusion_icon_circle bg-success bg-opacity-25">
|
|
<i class="fa fa-cubes text-success"/>
|
|
</div>
|
|
</div>
|
|
<div class="text-muted small mt-2">
|
|
<t t-esc="state.totalConsumers"/> total registered
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
|
|
<div class="col-md-3 col-sm-6">
|
|
<div class="card h-100 border-0 shadow-sm o_fusion_stat_card"
|
|
style="cursor:pointer" t-on-click="openUsageLog">
|
|
<div class="card-body">
|
|
<div class="d-flex justify-content-between align-items-start">
|
|
<div>
|
|
<div class="text-muted small text-uppercase">Month Cost</div>
|
|
<h2 class="mb-0 mt-1">$<t t-esc="formatCost(state.monthCost)"/></h2>
|
|
</div>
|
|
<div class="o_fusion_icon_circle bg-warning bg-opacity-25">
|
|
<i class="fa fa-usd text-warning"/>
|
|
</div>
|
|
</div>
|
|
<div class="text-muted small mt-2">
|
|
<t t-esc="state.monthRequests"/> requests this month
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
|
|
<div class="col-md-3 col-sm-6">
|
|
<div class="card h-100 border-0 shadow-sm o_fusion_stat_card"
|
|
style="cursor:pointer" t-on-click="openUsageLog">
|
|
<div class="card-body">
|
|
<div class="d-flex justify-content-between align-items-start">
|
|
<div>
|
|
<div class="text-muted small text-uppercase">Today</div>
|
|
<h2 class="mb-0 mt-1" t-esc="state.todayRequests"/>
|
|
</div>
|
|
<div class="o_fusion_icon_circle bg-info bg-opacity-25">
|
|
<i class="fa fa-bar-chart text-info"/>
|
|
</div>
|
|
</div>
|
|
<div class="text-muted small mt-2">
|
|
requests today
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
|
|
<!-- Budget Alerts -->
|
|
<div class="mb-4" t-if="state.approachingLimits.length > 0">
|
|
<div class="alert alert-warning d-flex align-items-start" t-foreach="state.approachingLimits" t-as="alert" t-key="alert_index">
|
|
<i class="fa fa-exclamation-triangle me-2 mt-1"/>
|
|
<div>
|
|
<strong t-esc="alert.consumer"/> on <t t-esc="alert.provider"/>:
|
|
<t t-esc="alert.pct"/>% of budget used
|
|
($<t t-esc="alert.spent"/> / $<t t-esc="alert.budget"/>)
|
|
</div>
|
|
</div>
|
|
</div>
|
|
|
|
<div class="row g-3">
|
|
<!-- Provider Stats -->
|
|
<div class="col-lg-6" t-if="state.providerStats.length > 0">
|
|
<div class="card border-0 shadow-sm h-100">
|
|
<div class="card-header bg-transparent border-bottom-0">
|
|
<h5 class="mb-0">Provider Activity</h5>
|
|
</div>
|
|
<div class="card-body p-0">
|
|
<table class="table table-hover mb-0">
|
|
<thead>
|
|
<tr>
|
|
<th>Provider</th>
|
|
<th class="text-end">Keys</th>
|
|
<th class="text-end">Requests</th>
|
|
<th class="text-end">Cost</th>
|
|
</tr>
|
|
</thead>
|
|
<tbody>
|
|
<tr t-foreach="state.providerStats" t-as="prov" t-key="prov.id">
|
|
<td><strong t-esc="prov.name"/></td>
|
|
<td class="text-end" t-esc="prov.keys"/>
|
|
<td class="text-end" t-esc="prov.requests"/>
|
|
<td class="text-end">$<t t-esc="formatCost(prov.cost)"/></td>
|
|
</tr>
|
|
</tbody>
|
|
</table>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
|
|
<!-- Top Consumers -->
|
|
<div class="col-lg-6" t-if="state.topConsumers.length > 0">
|
|
<div class="card border-0 shadow-sm h-100">
|
|
<div class="card-header bg-transparent border-bottom-0">
|
|
<h5 class="mb-0">Top Consumers (This Month)</h5>
|
|
</div>
|
|
<div class="card-body p-0">
|
|
<table class="table table-hover mb-0">
|
|
<thead>
|
|
<tr>
|
|
<th>Module</th>
|
|
<th class="text-end">Requests</th>
|
|
<th class="text-end">Cost</th>
|
|
</tr>
|
|
</thead>
|
|
<tbody>
|
|
<tr t-foreach="state.topConsumers" t-as="cons" t-key="cons_index">
|
|
<td><strong t-esc="cons.name"/></td>
|
|
<td class="text-end" t-esc="cons.requests"/>
|
|
<td class="text-end">$<t t-esc="formatCost(cons.cost)"/></td>
|
|
</tr>
|
|
</tbody>
|
|
</table>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
|
|
<!-- Recent Activity -->
|
|
<div class="mt-3" t-if="state.recentUsage.length > 0">
|
|
<div class="card border-0 shadow-sm">
|
|
<div class="card-header bg-transparent border-bottom-0 d-flex justify-content-between align-items-center">
|
|
<h5 class="mb-0">Recent Activity</h5>
|
|
<button class="btn btn-sm btn-outline-secondary" t-on-click="openUsageLog">
|
|
View All
|
|
</button>
|
|
</div>
|
|
<div class="card-body p-0">
|
|
<table class="table table-hover mb-0">
|
|
<thead>
|
|
<tr>
|
|
<th>Time</th>
|
|
<th>Consumer</th>
|
|
<th>Provider</th>
|
|
<th>Feature</th>
|
|
<th class="text-end">Tokens</th>
|
|
<th class="text-end">Cost</th>
|
|
<th>Status</th>
|
|
</tr>
|
|
</thead>
|
|
<tbody>
|
|
<tr t-foreach="state.recentUsage" t-as="usage" t-key="usage_index">
|
|
<td class="text-muted small" t-esc="formatTime(usage.time)"/>
|
|
<td t-esc="usage.consumer"/>
|
|
<td t-esc="usage.provider"/>
|
|
<td class="text-muted" t-esc="usage.feature"/>
|
|
<td class="text-end" t-esc="usage.tokens"/>
|
|
<td class="text-end">$<t t-esc="formatCostDetailed(usage.cost)"/></td>
|
|
<td>
|
|
<span class="badge" t-att-class="getStatusClass(usage.status)"
|
|
t-esc="usage.status"/>
|
|
</td>
|
|
</tr>
|
|
</tbody>
|
|
</table>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
|
|
<!-- Empty State -->
|
|
<div class="text-center py-5" t-if="state.monthRequests === 0 and state.activeProviders === 0">
|
|
<i class="fa fa-rocket fa-3x text-muted mb-3 d-block"/>
|
|
<h4>Welcome to Fusion API</h4>
|
|
<p class="text-muted mb-3">
|
|
Get started by configuring your API providers and adding your keys.
|
|
</p>
|
|
<button class="btn btn-primary" t-on-click="openProviders">
|
|
<i class="fa fa-plus me-1"/> Configure Providers
|
|
</button>
|
|
</div>
|
|
</div>
|
|
|
|
<!-- Loading State -->
|
|
<div class="text-center py-5" t-if="!state.loaded">
|
|
<i class="fa fa-spinner fa-spin fa-2x text-muted"/>
|
|
<p class="text-muted mt-2">Loading dashboard...</p>
|
|
</div>
|
|
</div>
|
|
</t>
|
|
</templates>
|