Files
html/tasks-live-opus5.html

99 lines
6.1 KiB
HTML

<!DOCTYPE html>
<html lang="fr"><head>
<meta charset="UTF-8"><title>WEVAL Tasks Live Opus5</title>
<style>
body{font-family:-apple-system,system-ui,sans-serif;background:#0a0e27;color:#e4e8f7;margin:0;padding:0;line-height:1.5}
.container{max-width:1400px;margin:0 auto;padding:24px}
h1{color:#6ba3ff;border-bottom:2px solid #1e3a8a;padding-bottom:8px;margin-top:0;font-size:22px;font-weight:500}
h2{color:#c084fc;margin-top:20px;font-size:18px;font-weight:500}
.grid{display:grid;grid-template-columns:repeat(auto-fit,minmax(260px,1fr));gap:16px;margin-top:16px}
.card{background:#141933;border:1px solid #263161;border-radius:10px;padding:16px}
.metric{font-size:26px;font-weight:500;margin:4px 0;color:#e4e8f7}
.metric-label{font-size:12px;color:#9ca8d3;text-transform:uppercase;letter-spacing:0.5px}
.status-ok{color:#10b981}.status-warn{color:#fbbf24}.status-bad{color:#ef4444}
.events{background:#0f1529;border:1px solid #263161;border-radius:10px;padding:16px;margin-top:20px;max-height:400px;overflow-y:auto}
.event-row{display:grid;grid-template-columns:75px 1fr 130px 70px;gap:10px;padding:8px 0;border-bottom:1px solid #1e2a4a;font-size:13px;align-items:center}
.event-row:last-child{border-bottom:none}
.event-time{color:#6b7a9e;font-family:SF Mono,monospace;font-size:11px}
.event-intent{color:#fbbf24;font-weight:500}
.event-prov{color:#9ca8d3;font-size:11px}
.event-ms{text-align:right;color:#9ca8d3;font-variant-numeric:tabular-nums}
.badge{display:inline-block;padding:3px 8px;border-radius:4px;font-size:10px;font-weight:500;text-transform:uppercase;letter-spacing:0.5px}
.badge-live{background:rgba(16,185,129,0.2);color:#10b981}
.badge-dead{background:rgba(239,68,68,0.2);color:#ef4444}
.trigger-btn{background:#1e3a8a;color:#e4e8f7;border:1px solid #2a4db5;padding:6px 12px;border-radius:6px;cursor:pointer;font-size:12px;margin:3px;transition:all 0.15s}
.trigger-btn:hover{background:#2a4db5;border-color:#3b5dcc}
.reply{background:#0a1224;padding:12px;border-radius:6px;margin-top:10px;font-family:SF Mono,monospace;font-size:12px;color:#d1d5db;max-height:200px;overflow-y:auto;white-space:pre-wrap}
</style>
</head><body>
<div class="container">
<h1>Tasks Live Opus5 Dispatch-Proxy Monitor</h1>
<p style="color:#9ca8d3;font-size:13px">Connected to /api/opus5-task-log.php auto-refresh 5s streaming ready</p>
<div class="grid">
<div class="card"><div class="metric-label">Events tracked</div><div class="metric" id="m-total"></div></div>
<div class="card"><div class="metric-label">Dispatches</div><div class="metric" id="m-dispatch"></div></div>
<div class="card"><div class="metric-label">Proxy (master-api)</div><div class="metric" id="m-proxy"></div></div>
<div class="card"><div class="metric-label">Avg latency</div><div class="metric" id="m-lat"></div></div>
<div class="card"><div class="metric-label">Ethica HCPs</div><div class="metric" id="m-ethica">146 694</div></div>
<div class="card"><div class="metric-label">NR status</div><div class="metric status-ok" id="m-nr">153/153</div></div>
<div class="card"><div class="metric-label">L99 status</div><div class="metric status-ok" id="m-l99">304/304</div></div>
<div class="card"><div class="metric-label">Blade</div><div class="metric" id="m-blade"><span class="badge badge-dead">DEAD 164h</span></div></div>
</div>
<h2>Quick triggers (one-click test)</h2>
<div style="margin-top:10px">
<button class="trigger-btn" onclick="trigger('sovereign status')">sovereign status</button>
<button class="trigger-btn" onclick="trigger('ethica live')">ethica live</button>
<button class="trigger-btn" onclick="trigger('s95 load')">s95 load</button>
<button class="trigger-btn" onclick="trigger('blade wake')">blade wake</button>
<button class="trigger-btn" onclick="trigger('p0 status')">p0 status</button>
<button class="trigger-btn" onclick="trigger('crm stats live')">crm stats</button>
<button class="trigger-btn" onclick="trigger('task log')">task log</button>
<button class="trigger-btn" onclick="trigger('combien de fichiers')">count 11435</button>
</div>
<div class="reply" id="reply" style="display:none"></div>
<h2>Recent events (live)</h2>
<div class="events" id="events"></div>
</div>
<script>
async function trigger(msg){
const reply=document.getElementById('reply');reply.style.display='block';reply.textContent='Loading: '+msg;
try{
const r=await fetch('/api/wevia-master-dispatch.php',{method:'POST',headers:{'Content-Type':'application/json'},body:JSON.stringify({message:msg,session:'tasks-ui'})});
const d=await r.json();
reply.textContent='['+d.provider+'] intent='+(d.intent||d.tool||'?')+' ms='+d.ms+'\n\n'+(d.response||d.content||d.output||JSON.stringify(d,null,2)).substring(0,2000);
setTimeout(load,500);
}catch(e){reply.textContent='Error: '+e.message}
}
async function load(){
try{
const r=await fetch('/api/opus5-task-log.php?limit=25');
const d=await r.json();
const events=d.events||[];
const stats=d.stats||{};
document.getElementById('m-total').textContent=(stats.total_lines||0).toLocaleString();
document.getElementById('m-dispatch').textContent=events.filter(e=>(e.provider||'').includes('dispatch-proxy')).length;
document.getElementById('m-proxy').textContent=events.filter(e=>e.type==='proxy').length;
const lats=events.map(e=>e.ms).filter(m=>typeof m==='number');
document.getElementById('m-lat').textContent=lats.length?Math.round(lats.reduce((a,b)=>a+b,0)/lats.length)+'ms':'';
const ev=document.getElementById('events');ev.innerHTML='';
events.slice(0,20).forEach(e=>{
const r=document.createElement('div');r.className='event-row';
const time=(e.ts||'').slice(11,19);
const intent=e.intent||'?';
const prov=(e.provider||'').replace('opus5-','').replace('opus-','');
const ms=e.ms?e.ms+'ms':'';
r.innerHTML='<span class="event-time">'+time+'</span><span class="event-intent">'+intent+'</span><span class="event-prov">'+prov+'</span><span class="event-ms">'+ms+'</span>';
ev.appendChild(r);
});
if(events.length===0){ev.innerHTML='<div style="color:#9ca8d3;padding:8px">No events yet. Click a trigger above.</div>';}
}catch(e){console.error(e)}
}
load();setInterval(load,5000);
</script>
</body></html>