468 lines
25 KiB
HTML
468 lines
25 KiB
HTML
<!DOCTYPE html>
|
|
<html lang="fr">
|
|
<head>
|
|
<meta charset="UTF-8">
|
|
<meta name="viewport" content="width=device-width,initial-scale=1">
|
|
<title>WEVIA Web IA Health — Command Center</title>
|
|
<script src="https://cdnjs.cloudflare.com/ajax/libs/Chart.js/4.4.0/chart.umd.min.js"></script>
|
|
<style>
|
|
:root{--bg:#0a0e1a;--bg2:#11162a;--bg3:#1a1f36;--fg:#e2e8f0;--dim:#94a3b8;--dim2:#64748b;
|
|
--teal:#00e5a0;--yellow:#f4c430;--red:#ef4444;--orange:#f59e0b;--purple:#a28fff;--cyan:#22d3ee;
|
|
--bd:rgba(255,255,255,0.08);--bd2:rgba(255,255,255,0.04);
|
|
--mf:"JetBrains Mono","SF Mono",monospace;--serif:"Playfair Display",Georgia,serif}
|
|
*{box-sizing:border-box;margin:0;padding:0}
|
|
body{background:var(--bg);color:var(--fg);font-family:-apple-system,BlinkMacSystemFont,"Segoe UI",Inter,system-ui,sans-serif;min-height:100vh;padding:16px;line-height:1.5;font-size:14px}
|
|
body::before{content:"";position:fixed;top:0;left:0;width:100%;height:100%;z-index:-1;background:radial-gradient(circle at 20% 10%,rgba(0,229,160,0.06),transparent 50%),radial-gradient(circle at 80% 90%,rgba(162,143,255,0.05),transparent 50%)}
|
|
.hdr{max-width:1500px;margin:0 auto 20px;display:flex;justify-content:space-between;align-items:center;flex-wrap:wrap;gap:14px}
|
|
.title{font-family:var(--serif);font-size:26px;font-weight:700;letter-spacing:-0.5px}
|
|
.title .accent{color:var(--teal)}
|
|
.meta{display:flex;gap:8px;flex-wrap:wrap;margin-top:6px}
|
|
.badge{padding:4px 10px;border-radius:999px;background:var(--bg2);border:1px solid var(--bd);font-family:var(--mf);font-size:10px;letter-spacing:0.4px}
|
|
.badge.ok{color:var(--teal);border-color:rgba(0,229,160,0.3)}
|
|
.badge.warn{color:var(--orange);border-color:rgba(245,158,11,0.3)}
|
|
.badge.err{color:var(--red);border-color:rgba(239,68,68,0.3)}
|
|
.grid{max-width:1500px;margin:0 auto;display:grid;grid-template-columns:repeat(12,1fr);gap:14px}
|
|
.card{background:var(--bg2);border:1px solid var(--bd);border-radius:14px;padding:18px;position:relative;overflow:hidden}
|
|
.card::before{content:"";position:absolute;top:0;left:0;right:0;height:2px;background:linear-gradient(90deg,var(--teal),var(--purple));opacity:0.4}
|
|
.col-12{grid-column:span 12}.col-8{grid-column:span 8}.col-6{grid-column:span 6}.col-4{grid-column:span 4}.col-3{grid-column:span 3}
|
|
@media(max-width:1024px){.col-8,.col-6,.col-4,.col-3{grid-column:span 12}}
|
|
.card-head{display:flex;justify-content:space-between;align-items:center;margin-bottom:14px;gap:10px}
|
|
.card-title{font-family:var(--serif);font-size:18px;font-weight:600}
|
|
.card-sub{font-family:var(--mf);font-size:9px;color:var(--dim2);letter-spacing:0.5px;text-transform:uppercase;margin-top:3px}
|
|
.big-stat{font-family:var(--serif);font-size:36px;font-weight:700;line-height:1;margin:4px 0}
|
|
.big-stat.ok{color:var(--teal)}.big-stat.warn{color:var(--orange)}.big-stat.err{color:var(--red)}
|
|
.kv{display:flex;justify-content:space-between;padding:7px 0;border-bottom:1px solid var(--bd2);font-family:var(--mf);font-size:11px}
|
|
.kv:last-child{border-bottom:none}.kv .k{color:var(--dim)}.kv .v{color:var(--fg);font-weight:600}
|
|
.rec{padding:9px 12px;border-radius:8px;background:var(--bg3);border-left:3px solid var(--teal);margin-top:8px;font-size:12px;line-height:1.5}
|
|
.rec.high{border-left-color:var(--red);background:rgba(239,68,68,0.08)}
|
|
.rec.med{border-left-color:var(--orange);background:rgba(245,158,11,0.06)}
|
|
.rec.low{border-left-color:var(--teal);background:rgba(0,229,160,0.06)}
|
|
.rec .lbl{font-family:var(--mf);font-size:8px;text-transform:uppercase;color:var(--dim);letter-spacing:1px;margin-bottom:3px}
|
|
.provider-grid{display:grid;grid-template-columns:repeat(4,1fr);gap:6px;margin-top:10px}
|
|
.provider-tile{background:var(--bg3);border:1px solid var(--bd2);border-radius:8px;padding:8px;text-align:center;cursor:pointer;transition:.2s}
|
|
.provider-tile:hover{border-color:var(--teal)}
|
|
.provider-tile .icon{font-size:16px}.provider-tile .name{font-family:var(--mf);font-size:9px;color:var(--dim);text-transform:uppercase;margin-top:3px}
|
|
.provider-tile.running{border-color:rgba(0,229,160,0.3);background:rgba(0,229,160,0.05)}
|
|
.provider-tile.running .name{color:var(--teal)}
|
|
.btn{padding:7px 12px;border-radius:8px;background:var(--bg3);border:1px solid var(--bd);color:var(--fg);cursor:pointer;font-family:var(--mf);font-size:10px;transition:.2s;text-decoration:none;display:inline-flex;align-items:center;gap:5px;letter-spacing:0.3px;white-space:nowrap}
|
|
.btn:hover{background:var(--teal);color:var(--bg);border-color:var(--teal)}
|
|
.btn.primary{background:var(--teal);color:var(--bg);font-weight:700}.btn.primary:hover{background:var(--yellow)}
|
|
.btn.danger{border-color:rgba(239,68,68,0.4);color:var(--red)}.btn.danger:hover{background:var(--red);color:#fff;border-color:var(--red)}
|
|
.btn.warn{border-color:rgba(245,158,11,0.4);color:var(--orange)}.btn.warn:hover{background:var(--orange);color:#000;border-color:var(--orange)}
|
|
.btn:disabled{opacity:0.5;cursor:wait}
|
|
.actions{display:flex;flex-wrap:wrap;gap:6px;margin-top:10px}
|
|
.chart-wrap{position:relative;height:180px;margin-top:10px}
|
|
.mini-chart{height:80px}
|
|
.gauge{width:120px;height:120px;margin:0 auto;position:relative}
|
|
.gauge svg{transform:rotate(-90deg)}
|
|
.gauge .track{stroke:var(--bg3);stroke-width:10;fill:none}
|
|
.gauge .fill{stroke-width:10;fill:none;stroke-linecap:round;transition:stroke-dashoffset .6s}
|
|
.gauge-label{position:absolute;top:50%;left:50%;transform:translate(-50%,-50%);text-align:center;font-family:var(--mf)}
|
|
.gauge-label .num{font-size:22px;font-weight:700}
|
|
.gauge-label .unit{font-size:9px;color:var(--dim);text-transform:uppercase;margin-top:2px}
|
|
.feed{max-height:280px;overflow-y:auto;margin-top:10px}
|
|
.feed-item{padding:8px 10px;border-radius:6px;background:var(--bg3);margin-bottom:5px;font-family:var(--mf);font-size:10px;display:flex;gap:8px;align-items:center}
|
|
.feed-item .st{width:60px;font-weight:700;text-align:center;padding:2px 4px;border-radius:3px}
|
|
.feed-item .st.done{color:var(--teal);background:rgba(0,229,160,0.1)}
|
|
.feed-item .st.failed{color:var(--red);background:rgba(239,68,68,0.1)}
|
|
.feed-item .st.failed_timeout{color:var(--red);background:rgba(239,68,68,0.1)}
|
|
.feed-item .st.dispatched{color:var(--orange);background:rgba(245,158,11,0.1)}
|
|
.feed-item .st.pending{color:var(--yellow);background:rgba(244,196,48,0.1)}
|
|
.feed-item .lbl{color:var(--dim);min-width:70px}
|
|
.feed-item .cmd{color:var(--fg);opacity:0.7;flex:1;overflow:hidden;text-overflow:ellipsis;white-space:nowrap}
|
|
.feed-item .age{color:var(--dim2);min-width:50px;text-align:right}
|
|
.ask-widget{background:var(--bg3);border-radius:10px;padding:12px;margin-top:10px}
|
|
.ask-row{display:flex;gap:6px;margin-bottom:6px;flex-wrap:wrap}
|
|
.ask-row select,.ask-row input{flex:1;min-width:120px;padding:8px 10px;border-radius:6px;background:var(--bg2);border:1px solid var(--bd);color:var(--fg);font-family:var(--mf);font-size:11px}
|
|
.ask-row textarea{width:100%;padding:8px 10px;border-radius:6px;background:var(--bg2);border:1px solid var(--bd);color:var(--fg);font-family:var(--mf);font-size:11px;resize:vertical;min-height:60px}
|
|
.ask-out{margin-top:8px;padding:10px;background:var(--bg);border-radius:6px;font-family:var(--mf);font-size:10px;color:var(--dim);max-height:200px;overflow-y:auto;white-space:pre-wrap;line-height:1.5;display:none}
|
|
.ask-out.show{display:block}
|
|
.checklist{margin-top:12px;padding:12px;background:var(--bg3);border-radius:10px;border:1px dashed var(--bd)}
|
|
.checklist .title{font-family:var(--mf);font-size:10px;text-transform:uppercase;color:var(--yellow);letter-spacing:0.7px;margin-bottom:8px}
|
|
.checklist ul{list-style:none;font-size:11px;line-height:1.8}
|
|
.checklist li::before{content:"▸ ";color:var(--teal)}
|
|
.status-pulse{display:inline-block;width:7px;height:7px;border-radius:50%;margin-right:5px;box-shadow:0 0 0 0 currentColor;animation:pulse 2s infinite}
|
|
@keyframes pulse{0%{box-shadow:0 0 0 0 currentColor}70%{box-shadow:0 0 0 7px transparent}100%{box-shadow:0 0 0 0 transparent}}
|
|
.footer{max-width:1500px;margin:24px auto 0;padding:14px 0;text-align:center;font-family:var(--mf);font-size:9px;color:var(--dim2);border-top:1px solid var(--bd)}
|
|
.toast{position:fixed;bottom:20px;right:20px;padding:12px 18px;border-radius:10px;background:var(--bg2);border:1px solid var(--teal);color:var(--teal);font-family:var(--mf);font-size:12px;box-shadow:0 8px 24px rgba(0,0,0,0.4);z-index:100;transition:transform .3s;transform:translateY(200%)}
|
|
.toast.show{transform:translateY(0)}
|
|
.toast.err{border-color:var(--red);color:var(--red)}
|
|
.loading{opacity:0.4;filter:blur(1px);pointer-events:none}
|
|
.metric-row{display:grid;grid-template-columns:repeat(4,1fr);gap:10px;margin-top:8px}
|
|
.metric-box{background:var(--bg3);padding:10px;border-radius:8px;text-align:center}
|
|
.metric-box .v{font-family:var(--mf);font-size:18px;font-weight:700;color:var(--teal)}
|
|
.metric-box .v.warn{color:var(--orange)}.metric-box .v.err{color:var(--red)}
|
|
.metric-box .l{font-family:var(--mf);font-size:9px;color:var(--dim);text-transform:uppercase;letter-spacing:0.4px;margin-top:2px}
|
|
</style>
|
|
</head>
|
|
<body>
|
|
|
|
<header class="hdr">
|
|
<div>
|
|
<div class="title"><span class="accent">WEVIA</span> · Command Center</div>
|
|
<div class="meta">
|
|
<span class="badge" id="b-blade">Blade: —</span>
|
|
<span class="badge" id="b-cdp">CDP: —</span>
|
|
<span class="badge" id="b-tasks">Tasks: —</span>
|
|
<span class="badge ok" id="b-intents">Intents: —</span>
|
|
<span class="badge" id="b-load">Load: —</span>
|
|
</div>
|
|
</div>
|
|
<div style="display:flex;gap:8px;align-items:center;flex-wrap:wrap">
|
|
<span style="font-family:var(--mf);font-size:9px;color:var(--dim2)">auto-refresh 30s</span>
|
|
<button class="btn" onclick="load()">↻ Refresh</button>
|
|
<a class="btn" href="/vnc-picker.html">VNC</a>
|
|
<a class="btn" href="/wevia-master.html">WEVIA Chat</a>
|
|
</div>
|
|
</header>
|
|
|
|
<div class="grid" id="grid">
|
|
<div class="card col-12" style="text-align:center;padding:40px;color:var(--dim)">Chargement...</div>
|
|
</div>
|
|
|
|
<div class="footer">
|
|
<div>WEVIA AUTONOMY v1.9 · phase 54 · doctrines 183-192 · <span id="f-ts">—</span></div>
|
|
<div style="margin-top:4px">Zero manuel · Auto-fallback · Auto-recovery 2min · Auto-harden 5min · 221 NL + 2367 opus4</div>
|
|
</div>
|
|
|
|
<div class="toast" id="toast"></div>
|
|
|
|
<script>
|
|
const GRID = document.getElementById("grid");
|
|
let chartTimeline, chartCoverage;
|
|
|
|
function toast(msg, err){
|
|
const t = document.getElementById("toast");
|
|
t.textContent = msg;
|
|
t.className = "toast show" + (err?" err":"");
|
|
setTimeout(()=>t.className="toast",4000);
|
|
}
|
|
|
|
async function load(){
|
|
GRID.classList.add("loading");
|
|
try{
|
|
const r = await fetch("/api/web-ia-health.php?_="+Date.now(),{cache:"no-store"});
|
|
const d = await r.json();
|
|
render(d);
|
|
}catch(e){
|
|
GRID.innerHTML = `<div class="card col-12"><div style="color:var(--red)">Erreur chargement: ${e.message}</div></div>`;
|
|
}
|
|
GRID.classList.remove("loading");
|
|
}
|
|
|
|
async function sendIntent(msg){
|
|
toast("Envoi: " + msg);
|
|
try{
|
|
const r = await fetch("/api/wevia-chat-v2-direct.php",{
|
|
method:"POST",headers:{"Content-Type":"application/json"},
|
|
body:JSON.stringify({message:msg,session_id:"dashboard-"+Date.now()})
|
|
});
|
|
const d = await r.json();
|
|
const out = d.content || d.response || d.answer || "(empty)";
|
|
toast("✓ " + (d.tool||d.intent||"OK") + " · " + Math.round(d.elapsed_ms||0) + "ms");
|
|
const modal = document.getElementById("askOut");
|
|
if (modal) {
|
|
modal.textContent = "=== " + msg + " ===\n\n" + out;
|
|
modal.classList.add("show");
|
|
}
|
|
setTimeout(load, 2000);
|
|
}catch(e){ toast("❌ "+e.message, true); }
|
|
}
|
|
|
|
function render(d){
|
|
const b = d.sections.blade || {};
|
|
const t = d.sections.tasks || {};
|
|
const cdp = d.sections.cdp_local || {};
|
|
const ints = d.sections.intents || {};
|
|
const recs = d.sections.recommendations || [];
|
|
const timeline = d.sections.tasks_timeline_24h || [];
|
|
const recent_tasks = d.sections.tasks_recent || [];
|
|
const recent_jobs = d.sections.jobs_recent || [];
|
|
const m = d.sections.s204 || {};
|
|
|
|
// Header badges
|
|
document.getElementById("b-blade").className = "badge " + (b.online?"ok":(b.heartbeat_age_s<600?"warn":"err"));
|
|
const bColorVar = b.color==='teal'?'var(--teal)':b.color==='orange'?'var(--orange)':'var(--red)';
|
|
document.getElementById("b-blade").innerHTML = `<span class="status-pulse" style="color:${bColorVar}"></span>Blade: ${b.status_label||'?'}`;
|
|
const cdp_ok = cdp.summary?cdp.summary.running:0;
|
|
const cdp_tot = cdp.summary?cdp.summary.total:8;
|
|
document.getElementById("b-cdp").className = "badge " + (cdp_ok>0?"ok":"warn");
|
|
document.getElementById("b-cdp").innerHTML = `CDP: ${cdp_ok}/${cdp_tot}`;
|
|
document.getElementById("b-tasks").innerHTML = `Tasks: ${t.done||0} done · ${t.stale||0} stale`;
|
|
document.getElementById("b-intents").innerHTML = `${ints.total||0} intents`;
|
|
const load5 = m.load?m.load["5m"]:0;
|
|
document.getElementById("b-load").className = "badge " + (load5<20?"ok":load5<50?"warn":"err");
|
|
document.getElementById("b-load").innerHTML = `Load: ${load5}`;
|
|
document.getElementById("f-ts").textContent = d.ts || "";
|
|
|
|
let html = "";
|
|
|
|
// === S204 METRICS STRIP ===
|
|
const disk_pct = m.disk?m.disk.pct:0;
|
|
const mem_pct = m.mem?m.mem.used_pct:0;
|
|
html += `
|
|
<div class="card col-12">
|
|
<div class="card-head">
|
|
<div><div class="card-title">📊 S204 Metrics (live)</div><div class="card-sub">Infrastructure serveur</div></div>
|
|
<div class="actions">
|
|
<button class="btn warn" onclick="sendIntent('disk_top')">Audit Disk</button>
|
|
<button class="btn warn" onclick="sendIntent('disaster_clean all')">Disaster Clean</button>
|
|
<button class="btn danger" onclick="sendIntent('deep_clean')">Deep Clean</button>
|
|
</div>
|
|
</div>
|
|
<div class="metric-row">
|
|
<div class="metric-box"><div class="v ${load5<20?'':load5<50?'warn':'err'}">${load5}</div><div class="l">Load 5min</div></div>
|
|
<div class="metric-box"><div class="v ${disk_pct<80?'':disk_pct<90?'warn':'err'}">${disk_pct}%</div><div class="l">Disk used</div></div>
|
|
<div class="metric-box"><div class="v ${mem_pct<70?'':mem_pct<85?'warn':'err'}">${mem_pct}%</div><div class="l">Mem used</div></div>
|
|
<div class="metric-box"><div class="v">${m.chromes||0}</div><div class="l">Chromes</div></div>
|
|
</div>
|
|
</div>`;
|
|
|
|
// === BLADE HERO ===
|
|
const hb_min = Math.floor((b.heartbeat_age_s||0)/60);
|
|
const hb_sec = (b.heartbeat_age_s||0) % 60;
|
|
const hb_txt = b.heartbeat_age_s<60 ? `${b.heartbeat_age_s}s` : (b.heartbeat_age_s<3600 ? `${hb_min}m ${hb_sec}s` : `${Math.floor(hb_min/60)}h ${hb_min%60}m`);
|
|
html += `
|
|
<div class="card col-8">
|
|
<div class="card-head">
|
|
<div><div class="card-title">🖥️ Agent Blade (PC Yacine)</div><div class="card-sub">Razer · Chrome cookies persistants · ZERO manuel</div></div>
|
|
<span class="badge ${b.online?'ok':(b.heartbeat_age_s<600?'warn':'err')}">${b.status_label||'?'}</span>
|
|
</div>
|
|
<div style="display:grid;grid-template-columns:1fr 1fr 1fr;gap:16px">
|
|
<div>
|
|
<div class="card-sub">Heartbeat</div>
|
|
<div class="big-stat ${b.online?'ok':(b.heartbeat_age_s<600?'warn':'err')}">${b.heartbeat_age_s>=0?hb_txt:'—'}</div>
|
|
<div class="card-sub" style="margin-top:4px">ago</div>
|
|
</div>
|
|
<div>
|
|
<div class="kv"><span class="k">IP</span><span class="v">${b.ip||'—'}</span></div>
|
|
<div class="kv"><span class="k">Agent</span><span class="v">v${b.agent_version||'—'}</span></div>
|
|
<div class="kv"><span class="k">Done</span><span class="v" style="color:var(--teal)">${t.done||0}</span></div>
|
|
<div class="kv"><span class="k">Running</span><span class="v" style="color:var(--orange)">${t.dispatched||0}</span></div>
|
|
<div class="kv"><span class="k">Stale</span><span class="v" style="color:var(--red)">${t.stale||0}</span></div>
|
|
</div>
|
|
<div>
|
|
<div class="rec ${b.online?'low':'high'}" style="margin-top:0">
|
|
<div class="lbl">RECO AUTO</div>${b.recommendation||'—'}
|
|
</div>
|
|
<div class="actions">
|
|
<button class="btn primary" onclick="sendIntent('blade_harden')">⚡ Harden</button>
|
|
<button class="btn" onclick="sendIntent('blade_tasks_recover')">↻ Recover</button>
|
|
<button class="btn" onclick="sendIntent('blade_health')">📊 Health</button>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
<div class="checklist">
|
|
<div class="title">Check-list PC Yacine · zéro manuel éternel</div>
|
|
<ul>
|
|
<li>Agent Blade v2.0 · Windows Task Scheduler au démarrage</li>
|
|
<li>Chrome ouvert foreground · sinon SendKeys échoue</li>
|
|
<li>Mode performance · pas économie d'énergie</li>
|
|
<li>Veille écran OK · pas veille hibernation</li>
|
|
<li>Auto-harden cron S204 toutes les 5min max 1x/24h</li>
|
|
</ul>
|
|
</div>
|
|
</div>`;
|
|
|
|
// === TIMELINE CHART ===
|
|
html += `
|
|
<div class="card col-4">
|
|
<div class="card-head">
|
|
<div><div class="card-title">📈 Tasks 24h</div><div class="card-sub">Timeline activité Blade</div></div>
|
|
</div>
|
|
<div class="chart-wrap"><canvas id="chartTimeline"></canvas></div>
|
|
</div>`;
|
|
|
|
// === CDP LOCAL WITH PROVIDERS ===
|
|
const provs = cdp.providers || [];
|
|
const coverage_pct = cdp.summary?cdp.summary.coverage_pct:0;
|
|
html += `
|
|
<div class="card col-6">
|
|
<div class="card-head">
|
|
<div><div class="card-title">🌐 CDP Local S204 (Fallback)</div><div class="card-sub">8 profiles Chrome · ports 9222-9229</div></div>
|
|
<span class="badge ${cdp_ok>0?'ok':'warn'}">${cdp_ok}/${cdp_tot}</span>
|
|
</div>
|
|
<div style="display:flex;gap:16px;align-items:center">
|
|
<div class="gauge">
|
|
<svg width="120" height="120" viewBox="0 0 120 120">
|
|
<circle class="track" cx="60" cy="60" r="50"/>
|
|
<circle class="fill" cx="60" cy="60" r="50" stroke="${coverage_pct>50?'var(--teal)':coverage_pct>0?'var(--orange)':'var(--red)'}" stroke-dasharray="${2*Math.PI*50}" stroke-dashoffset="${2*Math.PI*50*(1-coverage_pct/100)}"/>
|
|
</svg>
|
|
<div class="gauge-label"><div class="num">${coverage_pct}%</div><div class="unit">coverage</div></div>
|
|
</div>
|
|
<div style="flex:1">
|
|
<div class="provider-grid">`;
|
|
provs.forEach(p=>{
|
|
const running = p.status==='running';
|
|
html += `<div class="provider-tile ${running?'running':''}" onclick="sendIntent('ask_${p.slug==='openai'?'chatgpt':p.slug==='anthropic'?'claude':p.slug==='google'?'gemini':p.slug}_web bonjour')" title="Test ${p.name}">
|
|
<div class="icon">${running?'✅':'❌'}</div><div class="name">${p.slug}</div>
|
|
</div>`;
|
|
});
|
|
if (provs.length===0){
|
|
html += `<div style="grid-column:1/-1;text-align:center;color:var(--dim);padding:12px;font-size:11px">CDP API down · lancez launch_chromes_all</div>`;
|
|
}
|
|
html += `</div></div></div>
|
|
<div class="actions">
|
|
<button class="btn primary" onclick="sendIntent('launch_chromes_all')">🚀 Launch 8 Chromes</button>
|
|
<button class="btn" onclick="sendIntent('chromes_status')">Status</button>
|
|
<a class="btn" href="/vnc-picker.html">VNC Picker</a>
|
|
</div>
|
|
</div>`;
|
|
|
|
// === QUICK ASK WIDGET ===
|
|
html += `
|
|
<div class="card col-6">
|
|
<div class="card-head">
|
|
<div><div class="card-title">💬 Quick Ask Any IA</div><div class="card-sub">Envoyer prompt direct via WEVIA</div></div>
|
|
</div>
|
|
<div class="ask-widget">
|
|
<div class="ask-row">
|
|
<select id="askProvider">
|
|
<optgroup label="Via BLADE (PC Yacine, zero login)">
|
|
<option value="ask_blade_claude">Claude (Blade)</option>
|
|
<option value="ask_blade_chatgpt">ChatGPT (Blade)</option>
|
|
<option value="ask_blade_gemini">Gemini (Blade)</option>
|
|
<option value="ask_blade_deepseek">DeepSeek (Blade)</option>
|
|
<option value="ask_blade_mistral">Mistral (Blade)</option>
|
|
<option value="ask_blade_poe">Poe (Blade)</option>
|
|
<option value="ask_blade_perplexity">Perplexity (Blade)</option>
|
|
</optgroup>
|
|
<optgroup label="Via CDP Local S204 (fallback)">
|
|
<option value="ask_claude_web">Claude (CDP local)</option>
|
|
<option value="ask_chatgpt_web">ChatGPT (CDP local)</option>
|
|
<option value="ask_gemini_web">Gemini (CDP local)</option>
|
|
<option value="ask_deepseek_web">DeepSeek (CDP local)</option>
|
|
<option value="ask_mistral_web">Mistral (CDP local)</option>
|
|
</optgroup>
|
|
</select>
|
|
</div>
|
|
<div class="ask-row">
|
|
<textarea id="askPrompt" placeholder="Votre prompt..."></textarea>
|
|
</div>
|
|
<div class="actions">
|
|
<button class="btn primary" onclick="doAsk()">⚡ Send</button>
|
|
<button class="btn" onclick="document.getElementById('askOut').classList.remove('show')">Clear</button>
|
|
</div>
|
|
<div class="ask-out" id="askOut"></div>
|
|
</div>
|
|
</div>`;
|
|
|
|
// === RECENT TASKS FEED ===
|
|
html += `
|
|
<div class="card col-6">
|
|
<div class="card-head">
|
|
<div><div class="card-title">📋 Recent Tasks (10)</div><div class="card-sub">Historique Blade</div></div>
|
|
</div>
|
|
<div class="feed">`;
|
|
recent_tasks.forEach(t=>{
|
|
const age = t.age_s<60?`${t.age_s}s`:t.age_s<3600?`${Math.floor(t.age_s/60)}m`:`${Math.floor(t.age_s/3600)}h`;
|
|
html += `<div class="feed-item">
|
|
<span class="st ${t.status}">${t.status}</span>
|
|
<span class="lbl">${t.label}</span>
|
|
<span class="cmd">${(t.cmd||'').replace(/</g,'<')}</span>
|
|
<span class="age">${age}</span>
|
|
</div>`;
|
|
});
|
|
if (recent_tasks.length===0) html+=`<div style="color:var(--dim);font-size:11px;padding:10px">Aucune task récente</div>`;
|
|
html += `</div></div>`;
|
|
|
|
// === RECENT ASYNC JOBS ===
|
|
html += `
|
|
<div class="card col-6">
|
|
<div class="card-head">
|
|
<div><div class="card-title">⚙️ Async Jobs</div><div class="card-sub">Derniers résultats /tmp/wevia-job-*</div></div>
|
|
<button class="btn" onclick="sendIntent('job_list')">Refresh</button>
|
|
</div>
|
|
<div class="feed">`;
|
|
recent_jobs.forEach(j=>{
|
|
const age = j.age_s<60?`${j.age_s}s`:j.age_s<3600?`${Math.floor(j.age_s/60)}m`:`${Math.floor(j.age_s/3600)}h`;
|
|
html += `<div class="feed-item" style="flex-direction:column;align-items:stretch">
|
|
<div style="display:flex;justify-content:space-between">
|
|
<span style="color:var(--teal)">${j.id.substring(0,30)}...</span>
|
|
<span class="age">${age}</span>
|
|
</div>
|
|
<div style="color:var(--dim);font-size:9px;margin-top:4px;white-space:pre-wrap;max-height:60px;overflow:hidden">${(j.preview||'').replace(/</g,'<').substring(0,200)}</div>
|
|
</div>`;
|
|
});
|
|
if (recent_jobs.length===0) html+=`<div style="color:var(--dim);font-size:11px;padding:10px">Aucun job récent</div>`;
|
|
html += `</div></div>`;
|
|
|
|
// === CAPABILITIES + RECOMMENDATIONS ===
|
|
html += `
|
|
<div class="card col-6">
|
|
<div class="card-head">
|
|
<div><div class="card-title">🧠 Capabilités WEVIA</div><div class="card-sub">NL + opus4-wired</div></div>
|
|
<span class="badge ok">${ints.total||0} total</span>
|
|
</div>
|
|
<div class="kv"><span class="k">NL Priority</span><span class="v">${ints.nl_priority||0}</span></div>
|
|
<div class="kv"><span class="k">Opus4 wired</span><span class="v">${ints.opus4_wired||0}</span></div>
|
|
<div class="card-sub" style="margin-top:12px">Actions rapides</div>
|
|
<div class="actions">
|
|
<button class="btn" onclick="sendIntent('web_ia_health')">web_ia_health</button>
|
|
<button class="btn" onclick="sendIntent('blade_status')">blade_status</button>
|
|
<button class="btn" onclick="sendIntent('chromes_status')">chromes_status</button>
|
|
<button class="btn" onclick="sendIntent('job_list')">job_list</button>
|
|
<button class="btn" onclick="sendIntent('infra_sante')">infra</button>
|
|
<button class="btn" onclick="sendIntent('disk_top')">disk_top</button>
|
|
</div>
|
|
</div>`;
|
|
|
|
html += `
|
|
<div class="card col-6">
|
|
<div class="card-head">
|
|
<div><div class="card-title">💡 Recommandations Auto</div><div class="card-sub">Analyses WEVIA contextuelles</div></div>
|
|
</div>`;
|
|
if (recs.length>0){
|
|
recs.forEach(r=>{
|
|
html += `<div class="rec ${r.priority||'low'}">
|
|
<div class="lbl">${(r.priority||'info').toUpperCase()}</div>${r.text}
|
|
</div>`;
|
|
});
|
|
} else {
|
|
html += `<div style="color:var(--teal);padding:16px;text-align:center">✅ Tout optimal · aucune action requise</div>`;
|
|
}
|
|
html += `</div>`;
|
|
|
|
GRID.innerHTML = html;
|
|
|
|
// Render timeline chart
|
|
setTimeout(()=>{
|
|
const ctx = document.getElementById("chartTimeline");
|
|
if (ctx && window.Chart) {
|
|
if (chartTimeline) chartTimeline.destroy();
|
|
chartTimeline = new Chart(ctx, {
|
|
type: 'bar',
|
|
data: {
|
|
labels: timeline.map((_,i)=>(24-i-1)+'h'),
|
|
datasets: [
|
|
{label:'Done',data:timeline.map(b=>b.done),backgroundColor:'rgba(0,229,160,0.7)',stack:'x'},
|
|
{label:'Failed',data:timeline.map(b=>b.failed),backgroundColor:'rgba(239,68,68,0.7)',stack:'x'},
|
|
{label:'Pending',data:timeline.map(b=>b.pending),backgroundColor:'rgba(245,158,11,0.7)',stack:'x'}
|
|
]
|
|
},
|
|
options: {
|
|
responsive:true,maintainAspectRatio:false,
|
|
plugins:{legend:{labels:{color:'#94a3b8',font:{size:9}}}},
|
|
scales:{
|
|
x:{stacked:true,ticks:{color:'#64748b',font:{size:8}},grid:{color:'rgba(255,255,255,0.04)'}},
|
|
y:{stacked:true,ticks:{color:'#64748b',font:{size:9}},grid:{color:'rgba(255,255,255,0.04)'}}
|
|
}
|
|
}
|
|
});
|
|
}
|
|
},100);
|
|
}
|
|
|
|
function doAsk(){
|
|
const prov = document.getElementById("askProvider").value;
|
|
const prompt = document.getElementById("askPrompt").value.trim();
|
|
if (!prompt) { toast("Prompt vide", true); return; }
|
|
sendIntent(prov + " " + prompt);
|
|
}
|
|
|
|
load();
|
|
setInterval(load, 30000);
|
|
</script>
|
|
</body>
|
|
</html>
|