All checks were successful
CD Pipeline / deploy (push) Successful in 2m37s
統帥質疑:「那六頁的視覺方格 UI/UX 搞好了嗎?還有新增頁面嗎?」
回答:沒有,從 Phase 38 開始一直推遲。本 commit 補做。
I-1: 6 頁 base.html → ewoooc_base.html
- host_health / ai_calls_dashboard / budget / promotion_review /
quality_trend / ppt_audit_history 全改
- {% extends "base.html" %} → {% extends "ewoooc_base.html" %}
- {% block content %} → {% block ewooo_content %}
- 自動繼承:sidebar 240px / topbar 64px / fonts (Inter+JetBrains+Noto Sans TC)
/ ewoooc-tokens.css / ewoooc-shell.css / search box / 米色背景
I-2: _ewoooc_shell.html 加「AI 觀測」nav group
- 7 個項目:觀測台總覽 / 主機健康 / AI 呼叫 / 預算控管 /
RAG 晉升審核 / 反饋趨勢 / PPT 視覺審核
- 對應 active_page='obs_*',正確高亮
- 編號 07-13(系統管理改 14)
I-3: 新增頁面 /observability/ + /observability/overview
- routes/admin_observability_routes.py::observability_overview
- 單頁聚合 8 表跨 JOIN 的 KPI:
• 三主機 24h 在線率(host_health_probes,per host card)
• AI 呼叫 24h(ai_calls:total/tokens/cost/error rate/RAG hit/cache hit)
• 當月成本累計
• 預算告警(ratio ≥ alert_pct 自動列表)
• AIOps 7d(incidents + heal_logs:自癒成功率)
• MCP 24h(mcp_calls:tool 呼叫 + cache 率 + cost)
• RAG 學習 30d(learning_episodes:待審 + 晉升率)
• PPT 視覺審核 7d(ppt_audit_results:通過率)
• 6 大子頁入口卡(含一行說明)
- 對應 Phase 44 daily Telegram summary 的 web 版本
- 全部失敗安全(個別 query 失敗只跳過該卡,不擋整頁)
升級對應:
- UI 框架:base.html → ewoooc_base.html ✅(sidebar + topbar + token css 已生效)
- 設計憲法:8 卡片 + 8 表跨 JOIN 全景 + 一頁式總覽
- 入口:sidebar 7 項 + 觀測台首頁
- 資料表覆蓋:4 表(Phase 38)→ 8 表(Phase 45)
注意:完整 design token 重塑(Bootstrap class → --momo-* token / 焦糖橘)
留待後續 phase;本 commit 重點是「框架升級 + 新總覽頁」。
Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
207 lines
9.0 KiB
HTML
207 lines
9.0 KiB
HTML
{% extends "ewoooc_base.html" %}
|
||
|
||
{% block title %}AI 呼叫總覽{% endblock %}
|
||
|
||
{% block ewooo_content %}
|
||
<div class="container-fluid mt-3">
|
||
<h2 class="mb-3"><i class="fas fa-chart-bar me-2"></i>AI 呼叫總覽
|
||
<small class="text-muted">過去 {{ hours }} 小時</small>
|
||
</h2>
|
||
|
||
{% if error %}
|
||
<div class="alert alert-warning"><strong><i class="fas fa-exclamation-triangle me-1"></i></strong> {{ error }}</div>
|
||
{% endif %}
|
||
|
||
<!-- Phase 40 D-7: 一鍵觸發 Code Review (L2 自動化) -->
|
||
<div class="mb-3">
|
||
<button class="btn btn-warning btn-sm" onclick="triggerCodeReview()">
|
||
<i class="fas fa-microscope me-1"></i>觸發 Code Review Pipeline (5 step)
|
||
</button>
|
||
<small class="text-muted ms-2">
|
||
看到錯誤率飆高?一鍵觸發 Hermes→OpenClaw→EA→NemoTron 5 步審查最新 commit,
|
||
在背景執行,完成後 Telegram 通知。
|
||
</small>
|
||
</div>
|
||
|
||
<!-- 篩選 bar -->
|
||
<form method="get" class="row g-2 mb-3">
|
||
<div class="col-auto">
|
||
<select name="hours" class="form-select form-select-sm">
|
||
{% for h in [1, 6, 24, 72, 168] %}
|
||
<option value="{{ h }}" {% if hours == h %}selected{% endif %}>
|
||
{% if h < 24 %}{{ h }} 小時{% else %}{{ (h//24) }} 天{% endif %}
|
||
</option>
|
||
{% endfor %}
|
||
</select>
|
||
</div>
|
||
<div class="col-auto">
|
||
<select name="caller" class="form-select form-select-sm">
|
||
<option value="">全部呼叫端</option>
|
||
{% for c in callers %}
|
||
<option value="{{ c }}" {% if caller_filter == c %}selected{% endif %}>{{ c }}</option>
|
||
{% endfor %}
|
||
</select>
|
||
</div>
|
||
<div class="col-auto">
|
||
<select name="provider" class="form-select form-select-sm">
|
||
<option value="">全部供應商</option>
|
||
{% for p in ['gcp_ollama','ollama_secondary','ollama_111','gemini','claude','nim','openrouter','nim_via_elephant'] %}
|
||
<option value="{{ p }}" {% if provider_filter == p %}selected{% endif %}>{{ p }}</option>
|
||
{% endfor %}
|
||
</select>
|
||
</div>
|
||
<div class="col-auto">
|
||
<button class="btn btn-primary btn-sm">篩選</button>
|
||
</div>
|
||
</form>
|
||
|
||
<!-- 總覽 KPI -->
|
||
<div class="row g-2 mb-3">
|
||
<div class="col-lg-2 col-md-4 col-sm-6"><div class="card p-2"><small>呼叫次數</small><h4>{{ "{:,}".format(summary.total_calls or 0) }}</h4></div></div>
|
||
<div class="col-lg-2 col-md-4 col-sm-6"><div class="card p-2"><small>Token 用量</small><h4>{{ "{:,}".format(summary.total_tokens or 0) }}</h4></div></div>
|
||
<div class="col-lg-2 col-md-4 col-sm-6"><div class="card p-2"><small>成本 (USD)</small><h4>${{ "%.2f"|format(summary.total_cost or 0) }}</h4></div></div>
|
||
<div class="col-lg-2 col-md-4 col-sm-6"><div class="card p-2"><small>平均耗時</small><h4>{{ summary.avg_duration or 0 }} ms</h4></div></div>
|
||
<div class="col-lg-2 col-md-4 col-sm-6"><div class="card p-2"><small>RAG 命中</small><h4 class="text-success">{{ summary.rag_hits or 0 }}</h4></div></div>
|
||
<div class="col-lg-2 col-md-4 col-sm-6"><div class="card p-2"><small>錯誤次數</small><h4 class="{% if (summary.error_calls or 0) > 0 %}text-danger{% endif %}">{{ summary.error_calls or 0 }}</h4></div></div>
|
||
</div>
|
||
|
||
<!-- Phase 39 D-3: caller × RAG × MCP 編排矩陣 -->
|
||
{% if caller_richness %}
|
||
<div class="card mb-3">
|
||
<div class="card-header"><strong><i class="fas fa-network-wired me-2"></i>呼叫端 × RAG × MCP 編排矩陣</strong>
|
||
<small class="text-muted">資料來源:ai_calls × mcp_calls × rag_query_log({{ hours }}h 內呼叫 ≥ 5 次的 caller)</small>
|
||
</div>
|
||
<div class="card-body p-0">
|
||
<table class="table table-sm mb-0">
|
||
<thead class="table-light">
|
||
<tr>
|
||
<th>呼叫端</th>
|
||
<th class="text-end">總呼叫</th>
|
||
<th class="text-end">RAG 命中率</th>
|
||
<th class="text-end">MCP 編排率</th>
|
||
<th class="text-end">RAG 反饋分數</th>
|
||
<th class="text-end">反饋筆數</th>
|
||
</tr>
|
||
</thead>
|
||
<tbody>
|
||
{% for c in caller_richness %}
|
||
<tr>
|
||
<td><code>{{ c.caller }}</code></td>
|
||
<td class="text-end">{{ "{:,}".format(c.total_calls) }}</td>
|
||
<td class="text-end">
|
||
<strong class="{% if c.rag_hit_rate >= 50 %}text-success{% elif c.rag_hit_rate >= 20 %}text-warning{% else %}text-muted{% endif %}">
|
||
{{ "%.1f"|format(c.rag_hit_rate) }}%
|
||
</strong>
|
||
<small class="text-muted">({{ c.rag_hits }})</small>
|
||
</td>
|
||
<td class="text-end">
|
||
<strong class="{% if c.mcp_rate >= 30 %}text-info{% elif c.mcp_rate >= 10 %}text-warning{% endif %}">
|
||
{{ "%.1f"|format(c.mcp_rate) }}%
|
||
</strong>
|
||
<small class="text-muted">({{ c.mcp_orchestrated }})</small>
|
||
</td>
|
||
<td class="text-end">
|
||
{% if c.feedback_count > 0 %}
|
||
<strong class="{% if c.avg_rag_feedback >= 4 %}text-success{% elif c.avg_rag_feedback >= 3 %}text-warning{% else %}text-danger{% endif %}">
|
||
{{ "%.2f"|format(c.avg_rag_feedback) }}/5
|
||
</strong>
|
||
{% else %}
|
||
<small class="text-muted">—</small>
|
||
{% endif %}
|
||
</td>
|
||
<td class="text-end">{{ c.feedback_count }}</td>
|
||
</tr>
|
||
{% endfor %}
|
||
</tbody>
|
||
</table>
|
||
</div>
|
||
<div class="card-footer small text-muted">
|
||
<i class="fas fa-info-circle me-1"></i>
|
||
<strong>RAG 命中率</strong>:caller 在此期間有多少呼叫吃到 RAG 召回;
|
||
<strong>MCP 編排率</strong>:有多少呼叫透過 request_id 串接到 MCP tool;
|
||
<strong>反饋分數</strong>:RAG 召回後人工反饋(1-5)。
|
||
</div>
|
||
</div>
|
||
{% endif %}
|
||
|
||
<!-- by provider -->
|
||
<div class="card mb-3">
|
||
<div class="card-header"><strong>依供應商分組</strong></div>
|
||
<div class="card-body p-0">
|
||
<table class="table table-sm mb-0">
|
||
<thead class="table-light">
|
||
<tr><th>供應商</th><th class="text-end">呼叫數</th><th class="text-end">Token 用量</th><th class="text-end">成本 (USD)</th></tr>
|
||
</thead>
|
||
<tbody>
|
||
{% for row in by_provider %}
|
||
<tr>
|
||
<td><code>{{ row.provider }}</code></td>
|
||
<td class="text-end">{{ "{:,}".format(row.calls) }}</td>
|
||
<td class="text-end">{{ "{:,}".format(row.tokens) }}</td>
|
||
<td class="text-end">${{ "%.2f"|format(row.cost) }}</td>
|
||
</tr>
|
||
{% endfor %}
|
||
</tbody>
|
||
</table>
|
||
</div>
|
||
</div>
|
||
|
||
<!-- recent calls -->
|
||
<div class="card">
|
||
<div class="card-header"><strong>最近呼叫(最新 100 筆)</strong></div>
|
||
<div class="card-body p-0">
|
||
<table class="table table-sm table-striped mb-0" style="font-size: 0.85em;">
|
||
<thead class="table-light">
|
||
<tr>
|
||
<th>編號</th><th>時間</th><th>呼叫端</th><th>供應商</th><th>模型</th>
|
||
<th class="text-end">輸入</th><th class="text-end">輸出</th><th class="text-end">耗時 ms</th>
|
||
<th>狀態</th><th class="text-end">成本 $</th><th>標記</th>
|
||
</tr>
|
||
</thead>
|
||
<tbody>
|
||
{% for r in recent %}
|
||
<tr {% if r.status not in ['ok','cache_only'] %}class="table-warning"{% endif %}>
|
||
<td>{{ r.id }}</td>
|
||
<td><small>{{ r.called_at }}</small></td>
|
||
<td><code>{{ r.caller }}</code></td>
|
||
<td><small>{{ r.provider }}</small></td>
|
||
<td><small>{{ r.model[:25] }}</small></td>
|
||
<td class="text-end">{{ r.in_tokens }}</td>
|
||
<td class="text-end">{{ r.out_tokens }}</td>
|
||
<td class="text-end">{{ r.duration_ms }}</td>
|
||
<td><small>{{ r.status }}</small></td>
|
||
<td class="text-end">${{ "%.4f"|format(r.cost) }}</td>
|
||
<td>
|
||
{% if r.cache_hit %}<span class="badge bg-success">快取</span>{% endif %}
|
||
{% if r.rag_hit %}<span class="badge bg-info">RAG</span>{% endif %}
|
||
</td>
|
||
</tr>
|
||
{% endfor %}
|
||
</tbody>
|
||
</table>
|
||
</div>
|
||
</div>
|
||
|
||
<p class="text-muted mt-2"><small>
|
||
<i class="fas fa-robot me-1"></i>Operation Ollama-First v5.0 / Phase 40 — AI 呼叫總覽(含 RAG/MCP 編排矩陣)
|
||
</small></p>
|
||
</div>
|
||
|
||
<script>
|
||
async function triggerCodeReview() {
|
||
if (!confirm('觸發 Code Review Pipeline?\n\n會對最新 commit 跑 5 step 審查(Hermes 掃描 → OpenClaw 摘要 → EA 決策 → NemoTron 行動),背景執行。')) return;
|
||
try {
|
||
const r = await fetch('/observability/ai_calls/trigger_code_review', {method: 'POST'});
|
||
const d = await r.json();
|
||
if (d.ok) {
|
||
alert(`✅ ${d.message}\n\nPipeline ID: ${d.pipeline_id}\nCommit: ${d.commit_sha}\n變更檔案: ${d.changed_files_count} 個`);
|
||
} else {
|
||
alert('❌ ' + (d.error || '觸發失敗'));
|
||
}
|
||
} catch (e) {
|
||
alert('Error: ' + e);
|
||
}
|
||
}
|
||
</script>
|
||
{% endblock %}
|