|
<!DOCTYPE html> |
|
<html lang="zh"> |
|
|
|
<head> |
|
<meta charset="UTF-8"> |
|
<link rel="icon" type="image/x-icon" href="data:image/x-icon;,"> |
|
<meta name="viewport" content="width=device-width, initial-scale=1.0"> |
|
<title>请求日志查看</title> |
|
|
|
<link rel="stylesheet" href="/static/shared-styles.css"> |
|
<script src="/static/shared.js"></script> |
|
<style> |
|
|
|
.stats-grid { |
|
display: grid; |
|
grid-template-columns: repeat(auto-fit, minmax(200px, 1fr)); |
|
gap: 20px; |
|
margin-bottom: var(--spacing); |
|
} |
|
|
|
.stat-card { |
|
background: var(--card-background); |
|
padding: 20px; |
|
border-radius: var(--border-radius); |
|
box-shadow: 0 2px 8px rgba(0, 0, 0, 0.08); |
|
transition: all var(--transition-fast); |
|
border: 1px solid var(--border-color); |
|
} |
|
|
|
.stat-card:hover { |
|
transform: translateY(-2px); |
|
box-shadow: 0 4px 12px rgba(0, 0, 0, 0.12); |
|
} |
|
|
|
.stat-card h4 { |
|
margin: 0 0 8px 0; |
|
color: var(--primary-color); |
|
} |
|
|
|
.stat-value { |
|
font-size: 28px; |
|
font-weight: 600; |
|
color: var(--primary-color); |
|
margin-top: 4px; |
|
} |
|
|
|
.refresh-container { |
|
display: flex; |
|
justify-content: space-between; |
|
align-items: center; |
|
} |
|
|
|
.auto-refresh { |
|
display: flex; |
|
align-items: center; |
|
gap: 8px; |
|
background: var(--card-background); |
|
padding: 8px 16px; |
|
border-radius: var(--border-radius); |
|
border: 1px solid var(--border-color); |
|
} |
|
|
|
.modal { |
|
display: none; |
|
position: fixed; |
|
z-index: 1000; |
|
left: 0; |
|
top: 0; |
|
width: 100%; |
|
height: 100%; |
|
background-color: rgba(0, 0, 0, 0.4); |
|
overflow-y: hidden; |
|
} |
|
|
|
.modal-content { |
|
background-color: var(--card-background); |
|
margin: 5% auto; |
|
padding: 20px; |
|
border-radius: var(--border-radius); |
|
width: 90%; |
|
max-width: 800px; |
|
max-height: 85vh; |
|
overflow-y: auto; |
|
border: 1px solid var(--border-color); |
|
box-shadow: 0 8px 24px rgba(0, 0, 0, 0.15); |
|
} |
|
|
|
.close { |
|
float: right; |
|
cursor: pointer; |
|
font-size: 28px; |
|
} |
|
|
|
.info-button { |
|
padding: 6px 12px; |
|
font-size: 14px; |
|
border-radius: var(--border-radius); |
|
transition: all var(--transition-fast); |
|
background: var(--primary-color-alpha); |
|
color: var(--primary-color); |
|
border: 1px solid var(--primary-color); |
|
} |
|
|
|
.info-button:hover { |
|
background: var(--primary-color); |
|
color: white; |
|
} |
|
|
|
.message-table { |
|
width: 100%; |
|
border-collapse: collapse; |
|
margin-top: 10px; |
|
margin: 0; |
|
border: 1px solid var(--border-color); |
|
} |
|
|
|
.message-table th, |
|
.message-table td { |
|
padding: 12px 16px; |
|
border-bottom: 1px solid var(--border-color); |
|
transition: background-color var(--transition-fast); |
|
} |
|
|
|
.message-table td { |
|
word-break: break-word; |
|
} |
|
|
|
.message-table td:nth-child(2) { |
|
max-width: 600px; |
|
} |
|
|
|
.message-table td:first-child { |
|
width: 80px; |
|
white-space: nowrap; |
|
} |
|
|
|
.modal-header { |
|
display: flex; |
|
justify-content: space-between; |
|
align-items: center; |
|
margin-bottom: 15px; |
|
} |
|
|
|
.modal-header h3 { |
|
margin: 0; |
|
} |
|
|
|
.close { |
|
font-size: 24px; |
|
font-weight: bold; |
|
cursor: pointer; |
|
padding: 5px 10px; |
|
} |
|
|
|
.close:hover { |
|
color: var(--primary-color); |
|
} |
|
|
|
.usage-progress-container { |
|
margin: 16px 0; |
|
height: 8px; |
|
background-color: var(--border-color); |
|
border-radius: 4px; |
|
overflow: hidden; |
|
} |
|
|
|
.usage-progress-bar { |
|
height: 100%; |
|
width: 0%; |
|
transition: width 0.3s ease; |
|
background-color: var(--primary-color); |
|
border-radius: 4px; |
|
} |
|
|
|
|
|
.usage-progress-bar.low { |
|
background-color: #4caf50; |
|
|
|
} |
|
|
|
.usage-progress-bar.medium { |
|
background-color: #ff9800; |
|
|
|
} |
|
|
|
.usage-progress-bar.high { |
|
background-color: #f44336; |
|
|
|
} |
|
|
|
|
|
.token-info-tooltip { |
|
position: relative; |
|
display: inline-block; |
|
} |
|
|
|
.token-info-tooltip .tooltip-content { |
|
visibility: hidden; |
|
position: absolute; |
|
z-index: 1002; |
|
background-color: var(--card-background); |
|
padding: 12px 15px; |
|
border-radius: var(--border-radius); |
|
box-shadow: 0 4px 16px rgba(0, 0, 0, 0.15); |
|
width: 280px; |
|
left: 50%; |
|
transform: translateX(-50%); |
|
bottom: calc(100% + 8px); |
|
opacity: 0; |
|
transition: opacity 0.3s, visibility 0.3s; |
|
text-align: left; |
|
line-height: 1.6; |
|
border: 1px solid var(--border-color); |
|
pointer-events: none; |
|
} |
|
|
|
.token-info-tooltip:hover .tooltip-content { |
|
visibility: visible; |
|
opacity: 1; |
|
} |
|
|
|
|
|
.token-info-tooltip .tooltip-content::after { |
|
content: ''; |
|
position: absolute; |
|
top: 100%; |
|
left: 50%; |
|
margin-left: -8px; |
|
border-width: 8px; |
|
border-style: solid; |
|
border-color: var(--card-background) transparent transparent transparent; |
|
} |
|
|
|
|
|
.token-info-tooltip::after { |
|
content: ''; |
|
position: absolute; |
|
left: 50%; |
|
transform: translateX(-50%); |
|
bottom: 100%; |
|
width: 100%; |
|
height: 10px; |
|
background: transparent; |
|
} |
|
|
|
|
|
.token-info-tooltip .tooltip-info-row { |
|
display: flex; |
|
justify-content: space-between; |
|
margin: 2px 0; |
|
} |
|
|
|
.token-info-tooltip .tooltip-info-row .label { |
|
color: var(--text-secondary); |
|
margin-right: 10px; |
|
} |
|
|
|
.token-info-tooltip .tooltip-info-row .value { |
|
font-weight: 500; |
|
word-break: break-word; |
|
} |
|
|
|
|
|
.prompt-preview .tooltip-content { |
|
width: 320px; |
|
max-height: 200px; |
|
overflow-y: auto; |
|
overflow-x: hidden; |
|
white-space: pre-wrap; |
|
word-break: break-word; |
|
} |
|
|
|
.prompt-preview .tooltip-content .message-meta { |
|
font-size: 0.8em; |
|
color: var(--text-secondary); |
|
padding: 0; |
|
margin: 0 0 4px 0; |
|
} |
|
|
|
.prompt-preview .tooltip-content .last-message { |
|
font-size: 0.9em; |
|
line-height: 1.5; |
|
color: var(--text-primary); |
|
margin: 0; |
|
padding: 0; |
|
white-space: pre-wrap; |
|
word-break: break-word; |
|
} |
|
|
|
|
|
.prompt-preview .tooltip-content::-webkit-scrollbar { |
|
width: 6px; |
|
} |
|
|
|
.prompt-preview .tooltip-content::-webkit-scrollbar-thumb { |
|
background-color: var(--border-color); |
|
border-radius: 3px; |
|
} |
|
|
|
.prompt-preview .tooltip-content::-webkit-scrollbar-track { |
|
background-color: var(--card-background); |
|
} |
|
|
|
|
|
.table-container { |
|
border-radius: var(--border-radius); |
|
overflow: hidden; |
|
border: 1px solid var(--border-color); |
|
} |
|
|
|
#logsTable { |
|
position: relative; |
|
z-index: 1; |
|
} |
|
|
|
#logsTable th { |
|
position: sticky; |
|
top: 0; |
|
z-index: 2; |
|
background: var(--primary-color); |
|
white-space: nowrap; |
|
transition: background-color 0.2s ease; |
|
} |
|
|
|
#logsTable td { |
|
padding: 12px 16px; |
|
border-bottom: 1px solid var(--border-color); |
|
transition: background-color var(--transition-fast); |
|
} |
|
|
|
|
|
@media (max-width: 768px) { |
|
.stats-grid { |
|
grid-template-columns: repeat(2, 1fr); |
|
gap: 12px; |
|
} |
|
|
|
.stat-card { |
|
padding: 16px; |
|
} |
|
|
|
.stat-value { |
|
font-size: 24px; |
|
} |
|
|
|
.modal-content { |
|
margin: 2% auto; |
|
width: 95%; |
|
padding: 16px; |
|
} |
|
} |
|
|
|
|
|
#logsTable tr:hover td { |
|
background-color: var(--hover-color, rgba(0, 0, 0, 0.02)); |
|
} |
|
</style> |
|
</head> |
|
|
|
<body> |
|
<h1>请求日志查看</h1> |
|
|
|
<div class="container"> |
|
<div class="form-group"> |
|
<label>认证令牌:</label> |
|
<input type="password" id="authToken" placeholder="输入 AUTH_TOKEN"> |
|
</div> |
|
<div class="refresh-container"> |
|
<div class="button-group"> |
|
<button onclick="fetchLogs()">刷新日志</button> |
|
</div> |
|
<div class="auto-refresh"> |
|
<input type="checkbox" id="autoRefresh" checked> |
|
<label for="autoRefresh">自动刷新 (60秒)</label> |
|
</div> |
|
</div> |
|
</div> |
|
|
|
<div class="container"> |
|
<div class="stats-grid"> |
|
<div class="stat-card"> |
|
<h4>总请求数</h4> |
|
<div id="totalRequests" class="stat-value">-</div> |
|
</div> |
|
<div class="stat-card"> |
|
<h4>活跃请求数</h4> |
|
<div id="activeRequests" class="stat-value">-</div> |
|
</div> |
|
<div class="stat-card"> |
|
<h4>错误请求数</h4> |
|
<div id="errorRequests" class="stat-value">-</div> |
|
</div> |
|
<div class="stat-card"> |
|
<h4>最后更新</h4> |
|
<div id="lastUpdate" class="stat-value">-</div> |
|
</div> |
|
</div> |
|
|
|
<div class="table-container"> |
|
<table id="logsTable"> |
|
<thead> |
|
<tr> |
|
<th>序</th> |
|
<th>时间</th> |
|
<th>模型</th> |
|
<th>Token信息</th> |
|
<th>Prompt</th> |
|
<th>用时/首字</th> |
|
<th>流式响应</th> |
|
<th>状态</th> |
|
<th>错误信息</th> |
|
</tr> |
|
</thead> |
|
<tbody id="logsBody"></tbody> |
|
</table> |
|
</div> |
|
</div> |
|
|
|
<div id="message"></div> |
|
|
|
|
|
<div id="tokenModal" class="modal"> |
|
<div class="modal-content"> |
|
<div class="modal-header"> |
|
<h3>Token 详细信息</h3> |
|
<span class="close">×</span> |
|
</div> |
|
<table class="message-table"> |
|
<tr> |
|
<td>Token:</td> |
|
<td id="modalToken"></td> |
|
</tr> |
|
<tr> |
|
<td>校验和:</td> |
|
<td id="modalChecksum"></td> |
|
</tr> |
|
<tr> |
|
<td colspan="2" style="text-align: center; font-weight: bold; background-color: var(--border-color);"> |
|
用户信息 |
|
</td> |
|
</tr> |
|
<tr> |
|
<td>邮箱:</td> |
|
<td id="modalEmail"></td> |
|
</tr> |
|
<tr> |
|
<td>用户名:</td> |
|
<td id="modalName"></td> |
|
</tr> |
|
<tr> |
|
<td>用户ID:</td> |
|
<td id="modalId"></td> |
|
</tr> |
|
<tr> |
|
<td>更新时间:</td> |
|
<td id="modalUpdatedAt"></td> |
|
</tr> |
|
<tr> |
|
<td colspan="2" style="text-align: center; font-weight: bold; background-color: var(--border-color);"> |
|
会员信息 |
|
</td> |
|
</tr> |
|
<tr> |
|
<td>会员类型:</td> |
|
<td id="modalMemberType"></td> |
|
</tr> |
|
<tr> |
|
<td>支付ID:</td> |
|
<td id="modalPaymentId"></td> |
|
</tr> |
|
<tr> |
|
<td>试用剩余:</td> |
|
<td id="modalTrialDays"></td> |
|
</tr> |
|
<tr> |
|
<td colspan="2" style="text-align: center; font-weight: bold; background-color: var(--border-color);"> |
|
使用量统计 (最近30天) |
|
</td> |
|
</tr> |
|
<tr> |
|
<td>Premium models:</td> |
|
<td id="modalPremiumUsage"></td> |
|
</tr> |
|
<tr> |
|
<td>Standard models:</td> |
|
<td id="modalStandardUsage"></td> |
|
</tr> |
|
<tr> |
|
<td>Unknown models:</td> |
|
<td id="modalUnknownUsage"></td> |
|
</tr> |
|
</table> |
|
<div id="usageProgressContainer"></div> |
|
</div> |
|
</div> |
|
|
|
<div id="promptModal" class="modal"> |
|
<div class="modal-content"> |
|
<div class="modal-header"> |
|
<h3>对话内容</h3> |
|
<span class="close">×</span> |
|
</div> |
|
<div id="promptContent"></div> |
|
</div> |
|
</div> |
|
|
|
<script> |
|
let refreshInterval; |
|
|
|
function updateStats(data) { |
|
document.getElementById('totalRequests').textContent = data.total || 0; |
|
document.getElementById('activeRequests').textContent = data.active || 0; |
|
if (data.error) { |
|
document.getElementById('errorRequests').textContent = data.error; |
|
document.getElementById('errorRequests').parentElement.style.display = ''; |
|
} else { |
|
document.getElementById('errorRequests').parentElement.style.display = 'none'; |
|
} |
|
document.getElementById('lastUpdate').textContent = |
|
new Date(data.timestamp).toLocaleTimeString(); |
|
} |
|
|
|
function getProgressBarClass(percentage) { |
|
if (percentage <= 60) return 'low'; |
|
if (percentage <= 85) return 'medium'; |
|
return 'high'; |
|
} |
|
|
|
function formatMembershipType(type) { |
|
if (!type) return '-'; |
|
|
|
switch (type) { |
|
case 'free_trial': return 'Pro Trial'; |
|
case 'pro': return 'Pro'; |
|
case 'free': return 'Free'; |
|
case 'enterprise': return 'Business'; |
|
default: return type |
|
.split('_') |
|
.map(word => word.charAt(0).toUpperCase() + word.slice(1)) |
|
.join(' '); |
|
} |
|
} |
|
|
|
function showTokenModal(tokenInfo) { |
|
const modal = document.getElementById('tokenModal'); |
|
document.getElementById('modalToken').textContent = tokenInfo.token || '-'; |
|
document.getElementById('modalChecksum').textContent = tokenInfo.checksum || '-'; |
|
|
|
if (tokenInfo.profile) { |
|
const { user, stripe, usage } = tokenInfo.profile; |
|
|
|
|
|
document.getElementById('modalEmail').textContent = user.email || '-'; |
|
document.getElementById('modalName').textContent = user.name || '-'; |
|
document.getElementById('modalId').textContent = user.id || '-'; |
|
document.getElementById('modalUpdatedAt').textContent = user.updated_at ? new Date(user.updated_at).toLocaleString() : '-'; |
|
|
|
|
|
document.getElementById('modalMemberType').textContent = |
|
formatMembershipType(stripe.membership_type); |
|
document.getElementById('modalPaymentId').textContent = stripe.payment_id || '-'; |
|
document.getElementById('modalTrialDays').textContent = |
|
stripe.days_remaining_on_trial > 0 ? `${stripe.days_remaining_on_trial}天` : '-'; |
|
|
|
|
|
const container = document.getElementById('usageProgressContainer'); |
|
container.innerHTML = ''; |
|
|
|
const models = { |
|
'modalPremiumUsage': usage.premium, |
|
'modalStandardUsage': usage.standard, |
|
'modalUnknownUsage': usage.unknown |
|
}; |
|
|
|
Object.entries(models).forEach(([elementId, modelData]) => { |
|
const element = document.getElementById(elementId); |
|
if (modelData) { |
|
const { requests, tokens, max_requests } = modelData; |
|
|
|
if (max_requests) { |
|
const percentage = (requests / max_requests * 100).toFixed(1); |
|
element.textContent = `${requests}/${max_requests} requests (${percentage}%), ${tokens} tokens`; |
|
|
|
const progressDiv = document.createElement('div'); |
|
progressDiv.className = 'usage-progress-container'; |
|
const colorClass = getProgressBarClass(parseFloat(percentage)); |
|
progressDiv.innerHTML = `<div class="usage-progress-bar ${colorClass}" style="width: ${percentage}%"></div>`; |
|
container.appendChild(progressDiv); |
|
} else { |
|
element.textContent = `${requests} requests, ${tokens} tokens`; |
|
} |
|
} else { |
|
element.textContent = '-'; |
|
} |
|
}); |
|
} else { |
|
|
|
[ |
|
'modalEmail', |
|
'modalName', |
|
'modalId', |
|
'modalUpdatedAt', |
|
'modalMemberType', |
|
'modalPaymentId', |
|
'modalTrialDays', |
|
'modalPremiumUsage', |
|
'modalStandardUsage', |
|
'modalUnknownUsage' |
|
].forEach(id => document.getElementById(id).textContent = '-'); |
|
document.getElementById('usageProgressContainer').innerHTML = ''; |
|
} |
|
|
|
modal.style.display = 'block'; |
|
} |
|
|
|
function formatSimpleTokenInfo(tokenInfo) { |
|
if (!tokenInfo.profile) return '无用户信息'; |
|
|
|
const { user, stripe, usage } = tokenInfo.profile; |
|
const premiumUsage = usage.premium ? |
|
`${usage.premium.requests}/${usage.premium.max_requests}` : '-'; |
|
|
|
const rows = [ |
|
['邮箱', user.email || '-'], |
|
...(user.name ? [['用户名', user.name]] : []), |
|
['会员', formatMembershipType(stripe.membership_type)], |
|
['Premium', premiumUsage] |
|
]; |
|
|
|
return rows.map(([label, value]) => `<div class="tooltip-info-row"><span class="label">${label}:</span><span class="value">${value}</span></div>`).join(''); |
|
} |
|
|
|
function updateTable(data) { |
|
const tbody = document.getElementById('logsBody'); |
|
updateStats(data); |
|
|
|
tbody.innerHTML = data.logs.map(log => `<tr><td>${log.id}</td><td>${new Date(log.timestamp).toLocaleString()}</td><td>${log.model}</td><td><div class="token-info-tooltip"><button class="info-button" onclick='showTokenModal(${JSON.stringify(log.token_info)})'>查看详情<div class="tooltip-content">${formatSimpleTokenInfo(log.token_info)}</div></button></div></td><td>${log.prompt ?`<div class="token-info-tooltip prompt-preview"><button class="info-button" onclick="showPromptModal(decodeURIComponent('${encodeURIComponent(log.prompt).replace(/'/g, "\\'")}'))">查看对话<div class="tooltip-content">${formatPromptPreview(log.prompt)}</div></button></div>` :'-'}</td><td>${formatTiming(log.timing.total, log.timing.first)}</td><td>${log.stream ? '是' : '否'}</td><td>${log.status}</td><td>${log.error || '-'}</td></tr>`).join(''); |
|
} |
|
|
|
function formatTiming(total, first) { |
|
const formattedTotal = total.toFixed(2); |
|
const formattedFirst = first !== null && first !== undefined ? `${first.toFixed(2)}s` : '-'; |
|
return `${formattedTotal}s / ${formattedFirst}`; |
|
} |
|
|
|
function formatPromptPreview(promptStr) { |
|
try { |
|
const messages = parsePrompt(promptStr); |
|
if (!messages || messages.length === 0) { |
|
return '无对话内容'; |
|
} |
|
|
|
// 获取最后一条消息 |
|
const lastMessage = messages[messages.length - 1]; |
|
const roleLabels = { |
|
'system': '系统', |
|
'user': '用户', |
|
'assistant': '助手' |
|
}; |
|
|
|
return `<div class="message-meta">最后一条消息 (${roleLabels[lastMessage.role] || lastMessage.role}):</div><div class="last-message">${lastMessage.content.replace(/&/g, '&').replace(/</g, '<').replace(/>/g, '>').replace(/\n/g, '<br>')}</div>`; |
|
} catch (e) { |
|
console.error('预览对话内容失败:', e); |
|
return '无法解析对话内容'; |
|
} |
|
} |
|
|
|
async function fetchLogs() { |
|
const data = await makeAuthenticatedRequest('/logs'); |
|
if (data) { |
|
updateTable(data); |
|
showGlobalMessage('日志获取成功'); |
|
} |
|
} |
|
|
|
// 自动刷新控制 |
|
document.getElementById('autoRefresh').addEventListener('change', function (e) { |
|
if (e.target.checked) { |
|
refreshInterval = setInterval(fetchLogs, 60000); |
|
} else { |
|
clearInterval(refreshInterval); |
|
} |
|
}); |
|
|
|
// 页面加载完成后自动获取日志 |
|
document.addEventListener('DOMContentLoaded', () => { |
|
const authToken = getAuthToken(); |
|
if (authToken) { |
|
document.getElementById('authToken').value = authToken; |
|
fetchLogs(); |
|
} |
|
// 启动自动刷新 |
|
refreshInterval = setInterval(fetchLogs, 60000); |
|
}); |
|
|
|
// 初始化 token 处理 |
|
initializeTokenHandling('authToken'); |
|
|
|
// 添加清理逻辑 |
|
window.addEventListener('beforeunload', () => { |
|
if (refreshInterval) { |
|
clearInterval(refreshInterval); |
|
} |
|
}); |
|
|
|
// 添加模态框关闭逻辑 |
|
document.querySelectorAll('.modal .close').forEach(closeBtn => { |
|
closeBtn.onclick = function () { |
|
this.closest('.modal').style.display = 'none'; |
|
} |
|
}); |
|
|
|
window.onclick = function (event) { |
|
if (event.target.classList.contains('modal')) { |
|
event.target.style.display = 'none'; |
|
} |
|
} |
|
</script> |
|
</body> |
|
|
|
</html> |