Files
html/wevia-cortex.html
WEVIA 2b4a7a4ddc
Some checks failed
WEVAL NonReg / nonreg (push) Has been cancelled
auto-sync-all
2026-04-16 14:08:12 +02:00

129 lines
21 KiB
HTML

<!DOCTYPE html><html lang="fr"><head><meta charset="UTF-8"><meta name="viewport" content="width=device-width,initial-scale=1">
<title>WEVIA CORTEX</title>
<link href="https://fonts.googleapis.com/css2?family=Inter:wght@300;400;500;600;700&family=JetBrains+Mono:wght@400;500&display=swap" rel="stylesheet">
<style>
:root{--bg:#fff;--bg2:#f9f9f8;--bg3:#f1f0ef;--sbg:#f1f0ef;--shov:#e5e3df;--brd:#e5e3df;--brd2:#d4d1cc;--text:#1a1a1a;--text2:#6b6560;--dim:#9b9590;--accent:#c96442;--green:#16a34a;--blue:#2563eb;--amber:#d97706;--red:#dc2626;--codebg:#282c34;--codetxt:#abb2bf;--shadow:0 1px 3px rgba(0,0,0,.05);--r:12px}
.dark{--bg:#1a1a1a;--bg2:#262626;--bg3:#333;--sbg:#141414;--shov:#2a2a2a;--brd:#333;--brd2:#444;--text:#e5e5e5;--text2:#999;--dim:#666}
*{margin:0;padding:0;box-sizing:border-box}body{padding-top:24px;font-family:'Inter',-apple-system,sans-serif;height:100vh;display:flex;color:var(--text);background:var(--bg)}
.sb{width:180px;display:none;background:var(--sbg);flex-direction:column;border-right:1px solid var(--brd);flex-shrink:0}
.sb-h{padding:12px}.nb{width:100%;display:flex;align-items:center;gap:8px;padding:10px 14px;border-radius:10px;border:1px solid var(--brd);background:var(--bg);font-size:14px;font-weight:500;color:var(--text);cursor:pointer}.nb:hover{background:var(--bg3)}
.sb-l{flex:1;overflow-y:auto;padding:4px 8px}.ci{padding:10px 12px;border-radius:8px;font-size:13px;color:var(--text2);cursor:pointer;margin-bottom:2px;white-space:nowrap;overflow:hidden;text-overflow:ellipsis}.ci:hover,.ci.a{background:var(--shov);color:var(--text)}
.sb-f{padding:12px;border-top:1px solid var(--brd);display:flex;gap:6px}.sb-f button{flex:1;padding:8px;border-radius:8px;border:1px solid var(--brd);background:transparent;font-size:11px;color:var(--dim);cursor:pointer}.sb-f button:hover{background:var(--shov)}
.main{flex:1;display:flex;flex-direction:column;overflow:hidden}
header{padding:8px 16px;display:flex;align-items:center;justify-content:center;gap:12px;border-bottom:1px solid var(--brd);background:var(--bg2);position:relative}
.mp{display:flex;align-items:center;gap:6px;padding:6px 14px;border-radius:20px;border:1px solid var(--brd);font-size:13px;font-weight:500}.mp .dot{width:7px;height:7px;border-radius:50%;background:var(--green)}
.pill{padding:5px 12px;border-radius:20px;border:1px solid var(--brd);font-size:12px;font-weight:600;cursor:pointer;transition:.2s;user-select:none}.pill:hover{background:var(--bg3)}
.pill.on{background:linear-gradient(135deg,#f59e0b,#f97316);color:#fff;border-color:transparent;box-shadow:0 2px 8px rgba(249,115,22,.3)}
.pill.ao{background:linear-gradient(135deg,var(--blue),#7c3aed);color:#fff;border-color:transparent;box-shadow:0 2px 8px rgba(37,99,235,.3)}
.hp{font-family:'JetBrains Mono',monospace;font-size:11px;color:var(--dim);position:absolute;right:16px}
.wel{flex:1;display:flex;flex-direction:column;align-items:center;justify-content:center;gap:16px}
.wel h1{font-size:28px;font-weight:700}.wel p{font-size:15px;color:var(--dim)}
.sugs{display:grid;grid-template-columns:1fr 1fr;gap:10px;max-width:500px;width:100%;padding:0 20px}
.sug{padding:14px 16px;border:1px solid var(--brd);border-radius:var(--r);cursor:pointer;transition:.15s}.sug:hover{background:var(--bg3);border-color:var(--brd2)}
.sug b{font-size:14px;display:block;margin-bottom:4px}.sug span{font-size:12px;color:var(--dim)}
.ca{flex:1;overflow-y:auto;display:flex;flex-direction:column}.msgs{max-width:720px;margin:0 auto;padding:20px 20px 120px;width:100%}
.msg{margin-bottom:20px;animation:fi .25s ease}@keyframes fi{from{opacity:0;transform:translateY(6px)}to{opacity:1;transform:none}}
.mh{display:flex;align-items:center;gap:8px;margin-bottom:6px}
.av{width:24px;height:24px;border-radius:50%;display:flex;align-items:center;justify-content:center;font-size:11px;font-weight:700;flex-shrink:0}
.msg.user .av{background:rgba(37,99,235,.1);color:var(--blue)}.msg.assistant .av{background:linear-gradient(135deg,#c96442,#d4845a);color:#fff}
.mn{font-size:13px;font-weight:600}.mt{font-size:11px;color:var(--dim);font-family:'JetBrains Mono',monospace}.mv{font-size:11px;color:var(--green);font-family:'JetBrains Mono',monospace}
.mb{font-size:15px;line-height:1.7;padding-left:32px}.mb p{margin-bottom:8px}
.mb pre{background:var(--codebg);color:var(--codetxt);border-radius:8px;padding:14px;margin:10px 0;overflow-x:auto;font-family:'JetBrains Mono',monospace;font-size:12px;line-height:1.5;position:relative}
.mb code{font-family:'JetBrains Mono',monospace;font-size:13px;background:var(--bg3);padding:2px 6px;border-radius:4px}.mb strong{font-weight:600}
.mb h2{font-size:17px;margin:16px 0 8px;font-weight:700}.mb h3{font-size:15px;margin:12px 0 6px;font-weight:600}
.mb ul{margin:8px 0 8px 20px}.mb li{margin-bottom:4px}.mb blockquote{border-left:3px solid var(--brd2);padding-left:12px;color:var(--dim);margin:8px 0}
.ts{background:var(--bg2);border:1px solid var(--brd);border-radius:8px;margin:6px 0 6px 32px;overflow:hidden;cursor:pointer}
.tsh{display:flex;align-items:center;gap:8px;padding:8px 12px;font-size:12px;font-family:'JetBrains Mono',monospace}
.tsh b{color:var(--blue)}.tsi{flex:1;color:var(--dim);font-size:11px;overflow:hidden;text-overflow:ellipsis;white-space:nowrap}
.tso{display:none;padding:0 12px 10px}.ts.open .tso{display:block}
.tso pre{background:var(--codebg);color:var(--codetxt);border-radius:6px;padding:10px;font-size:11px;max-height:200px;overflow:auto;white-space:pre-wrap;margin:0}
.ld{display:none;padding:0 20px}.ld.on{display:block}.ldi{max-width:720px;margin:0 auto;padding-left:32px;display:flex;align-items:center;gap:10px}
.dots{display:flex;gap:3px}.dots span{width:5px;height:5px;border-radius:50%;background:var(--accent);animation:dp .6s infinite alternate}.dots span:nth-child(2){animation-delay:.15s}.dots span:nth-child(3){animation-delay:.3s}@keyframes dp{from{opacity:.2;transform:scale(.8)}to{opacity:1;transform:scale(1)}}
.lt{font-size:12px;color:var(--dim);font-style:italic}
.iw{position:fixed;bottom:0;left:0;right:0;background:linear-gradient(0deg,var(--bg) 80%,transparent);padding:12px 20px 16px}
.ib{max-width:720px;margin:0 auto;background:var(--bg2);border:1px solid var(--brd);border-radius:16px;box-shadow:var(--shadow),0 4px 12px rgba(0,0,0,.04);overflow:hidden;transition:border .2s}
.ib:focus-within{border-color:var(--accent);box-shadow:0 0 0 2px rgba(201,100,66,.15)}
.ir{display:flex;align-items:flex-end;padding:4px}
.ir textarea{flex:1;resize:none;border:none;outline:none;padding:10px 14px;font-size:15px;font-family:'Inter',sans-serif;color:var(--text);background:transparent;max-height:160px;line-height:1.5}
.ir textarea::placeholder{color:var(--dim)}
.sn{width:36px;height:36px;border-radius:10px;border:none;background:var(--text);color:var(--bg2);cursor:pointer;display:flex;align-items:center;justify-content:center;margin:2px;flex-shrink:0}
.sn:hover{opacity:.8}.sn:disabled{opacity:.2;cursor:default}.sn svg{width:16px;height:16px}
.if{display:flex;justify-content:space-between;padding:0 14px 8px;font-size:11px;color:var(--dim)}
dialog{border:1px solid var(--brd);border-radius:var(--r);background:var(--bg2);color:var(--text);padding:20px;max-width:400px;width:90%}dialog::backdrop{background:rgba(0,0,0,.3)}
dialog h3{font-size:14px;margin-bottom:12px}dialog textarea{width:100%;height:80px;resize:vertical;background:var(--bg);border:1px solid var(--brd);border-radius:8px;padding:8px;font-size:12px;font-family:'JetBrains Mono',monospace;color:var(--text);outline:none}
.dr{display:flex;align-items:center;gap:8px;margin-top:10px}.dr label{font-size:12px;color:var(--text2);min-width:60px}.dr input[type=range]{flex:1;accent-color:var(--accent)}.dr .v{font-family:'JetBrains Mono',monospace;font-size:11px;color:var(--accent);min-width:30px}
@media(max-width:768px){.sb{display:none}.iw{left:0}}
</style><style>#wnav{display:none!important}</style></head><body><div id="live-stats" ondblclick="this.remove()" style="position:fixed;top:0;left:0;right:0;z-index:9999;display:flex;justify-content:center;gap:12px;padding:4px 8px;background:linear-gradient(135deg,#1e293b,#0f172a);font-family:sans-serif"><div style="color:#4ade80;font:700 10px sans-serif"></head><body>#9889; <span id="ls-ag">669</span> Agents</div><div style="color:#60a5fa;font:700 10px sans-serif"></head><body>#127970; <span id="ls-dp">22</span> Depts</div><div style="color:#fbbf24;font:700 10px sans-serif"></head><body>#128051; 20 Docker</div><div style="color:#a78bfa;font:700 10px sans-serif"></head><body>#129302; 10 Ollama</div><div style="color:#f87171;font:700 10px sans-serif"></head><body>#128200; <span id="ls-nr">152/153</span></div><div style="color:#34d399;font:700 10px sans-serif"></head><body>#128274; SSO OK</div><div style="width:6px;height:6px;border-radius:50%;background:#4ade80;animation:lp 2s infinite;align-self:center"></div></div><style>@keyframes lp{0%,100%{opacity:1}50%{opacity:.3}}</style><div id="wnav" style="position:relative;z-index:100;padding:8px 12px;display:flex;justify-content:center;gap:6px;background:transparent;font-family:sans-serif"><a href="/l99-saas.html" style="padding:4px 12px;border-radius:6px;font-size:10px;font-weight:600;text-decoration:none;background:rgba(0,0,0,.06);color:#555;transition:all .2s">L99</a><a href="/admin-saas.html" style="padding:4px 12px;border-radius:6px;font-size:10px;font-weight:600;text-decoration:none;background:rgba(0,0,0,.06);color:#555;transition:all .2s">Admin</a><a href="/realtime-monitor.html" style="padding:4px 12px;border-radius:6px;font-size:10px;font-weight:600;text-decoration:none;background:rgba(0,0,0,.06);color:#555;transition:all .2s">Monitor</a><a href="/agents-goodjob.html" style="padding:4px 12px;border-radius:6px;font-size:10px;font-weight:600;text-decoration:none;background:rgba(0,0,0,.06);color:#555;transition:all .2s">Enterprise</a><a href="/sovereign-claude.html" style="padding:4px 12px;border-radius:6px;font-size:10px;font-weight:600;text-decoration:none;background:rgba(0,0,0,.06);color:#555;transition:all .2s">Sovereign</a><a href="/cyber-monitor.html" style="padding:4px 12px;border-radius:6px;font-size:10px;font-weight:600;text-decoration:none;background:rgba(0,0,0,.06);color:#555;transition:all .2s">Cyber</a></div>
56a
<div class="sb"><div class="sb-h"><button class="nb" onclick="newChat()">+ Nouveau chat</button></div><div class="sb-l" id="sbl"></div><div class="sb-f"><button onclick="document.body.classList.toggle('dark')">◐ Thème</button><button onclick="document.getElementById('cfg').showModal()">⚙ Config</button></div></div>
<div class="main">
<header><div class="mp"><span class="dot" id="hd"></span><span>WEVIA CORTEX v2.1.89</span></div><div class="pill on" id="tp" onclick="turbo=!turbo;this.classList.toggle('on',turbo);updL()">⚡ Turbo</div><div class="pill ao" id="ap" onclick="agent=!agent;this.classList.toggle('ao',agent);updL()">🤖 Agent</div><span class="hp" id="hp">...</span></header>
<div class="wel" id="wel"><h1>Bonjour, Yanis</h1><p>Comment puis-je vous aider aujourd'hui ?</p><div class="sugs"><div class="sug" onclick="ask('Audit infrastructure: Analyse l\'état complet de S204, S95 et S151')"><b>Audit infrastructure</b><span>État de S204, S95, S151</span></div><div class="sug" onclick="ask('Status Ethica: Combien de HCPs par pays, derniers enrichissements')"><b>Status Ethica</b><span>HCPs, enrichissements</span></div><div class="sug" onclick="ask('Architecture WEVAL: Explique la stack technique complète avec tous les services')"><b>Architecture WEVAL</b><span>Stack technique complète</span></div><div class="sug" onclick="ask('NonReg: Lance les tests et montre les résultats')"><b>NonReg status</b><span>Derniers tests NonReg</span></div></div></div>
<div class="ca" id="ca"><div class="msgs" id="msgs"></div></div>
<div class="ld" id="ld"><div class="ldi"><div class="dots"><span></span><span></span><span></span></div><span class="lt" id="ldt">Réflexion...</span></div></div>
<div class="iw"><div class="ib"><div class="ir"><textarea id="inp" placeholder="Message à WEVIA CORTEX..." rows="1" onkeydown="if(event.key==='Enter'&&!event.shiftKey){event.preventDefault();send()}" oninput="this.style.height='auto';this.style.height=Math.min(this.scrollHeight,160)+'px'"></textarea><button class="sn" id="sb" onclick="send()"><svg viewBox="0 0 16 16" fill="currentColor"><path d="M1.5 1.5l13 6.5-13 6.5V9l8-1-8-1z"/></svg></button></div><div class="if"><span id="ml">Turbo ⚡ + Agent 🤖</span><span id="tc"></span></div></div></div>
</div>
<dialog id="cfg"><h3>Configuration</h3><textarea id="sys">Tu es l'assistant IA souverain de WEVAL Consulting. Direct, technique, expert en ERP/SAP, cloud, cybersécurité et IA. En mode Agent, utilise les outils (bash, read_file, sentinel, api_call) pour scanner et corriger l'infrastructure. Réponds en français.</textarea><div class="dr"><label>Temp</label><input type="range" min="0" max="100" value="10" oninput="document.getElementById('tv').textContent=(this.value/100).toFixed(2)"><span class="v" id="tv">0.10</span></div><div class="dr"><label>Max tok</label><input type="range" min="100" max="4096" value="2048" step="100" oninput="document.getElementById('mtv').textContent=this.value"><span class="v" id="mtv">2048</span></div><div class="dr" style="margin-top:16px"><button onclick="document.getElementById('cfg').close()" style="width:100%;padding:10px;border-radius:8px;border:none;background:var(--text);color:var(--bg);font-size:13px;font-weight:600;cursor:pointer">Fermer</button></div></dialog>
<script>
const API='/api/wevia-stream-api.php';
const API_LEGACY='/api/wevia-sovereign-proxy.php';
let H=[],turbo=true,agent=true,gen=false,ttok=0,nreq=0;
let chats=[{id:1,t:'Nouveau chat',m:[]}],ci=0;
function updL(){document.getElementById('ml').textContent=(turbo?'Turbo ⚡':'Local 🏠')+' + '+(agent?'Agent 🤖':'Chat 💬')}
function newChat(){chats.push({id:Date.now(),t:'Nouveau chat',m:[]});ci=chats.length-1;H=[];document.getElementById('msgs').innerHTML='';document.getElementById('wel').style.display='flex';document.getElementById('ca').style.display='none';rSb()}
function rSb(){document.getElementById('sbl').innerHTML=chats.map((c,i)=>'<div class="ci'+(i===ci?' a':'')+'" onclick="swC('+i+')">'+c.t+'</div>').join('')}
function swC(i){ci=i;H=chats[i].m.map(m=>({role:m.r,content:m.t}));var el=document.getElementById('msgs');el.innerHTML='';chats[i].m.forEach(function(m){rMsg(m.r,m.t,m.o||{})});document.getElementById('wel').style.display=chats[i].m.length?'none':'flex';document.getElementById('ca').style.display=chats[i].m.length?'flex':'none';rSb()}
function ask(t){document.getElementById('inp').value=t;send()}
function esc(s){return s.replace(/&/g,'&amp;').replace(/</g,'&lt;').replace(/>/g,'&gt;')}
function md(t){t=t.replace(/```(\w*)\n?([\s\S]*?)```/g,function(m,l,c){var id='c'+Math.random().toString(36).slice(2,7);return'<pre><code id="'+id+'">'+esc(c.trim())+'</code></pre>'});t=t.replace(/`([^`]+)`/g,'<code>$1</code>');t=t.replace(/\*\*([^*]+)\*\*/g,'<strong>$1</strong>');t=t.replace(/^### (.+)$/gm,'<h3>$1</h3>');t=t.replace(/^## (.+)$/gm,'<h2>$1</h2>');t=t.replace(/^\- (.+)$/gm,'<li>$1</li>');t=t.replace(/(<li>[\s\S]*?<\/li>)/g,'<ul>$1</ul>');t=t.replace(/<\/ul>\s*<ul>/g,'');t=t.replace(/^> (.+)$/gm,'<blockquote>$1</blockquote>');t=t.replace(/\n\n/g,'</p><p>');return'<p>'+t+'</p>'}
function rMsg(role,content,o){o=o||{};var el=document.getElementById('msgs'),tm=new Date().toLocaleTimeString('fr',{hour:'2-digit',minute:'2-digit'});
var steps='';if(o.steps&&o.steps.length){steps=o.steps.map(function(s){var ic={bash:'>_',read_file:'F',edit_file:'E',sentinel:'S',api_call:'A'}[s.tool]||'?';return'<div class="ts" onclick="this.classList.toggle(\'open\')"><div class="tsh"><b>['+ic+'] '+s.tool+'</b><span class="tsi">'+esc(JSON.stringify(s.input).slice(0,60))+'</span><span>'+(s.is_error?'ERR':'OK')+'</span></div><div class="tso"><pre>'+esc((s.output||'').slice(0,800))+'</pre></div></div>'}).join('')}
var via=o.provider?'<span class="mv">via '+o.provider+'</span>':'';var lat=o.latency?'<span class="mt"> '+o.latency+'</span>':'';
el.innerHTML+='<div class="msg '+role+'"><div class="mh"><div class="av">'+(role==='user'?'Y':'S')+'</div><span class="mn">'+(role==='user'?'Vous':'WEVIA CORTEX')+'</span><span class="mt">'+tm+'</span>'+via+lat+'</div>'+steps+'<div class="mb">'+(role==='user'?'<p>'+esc(content)+'</p>':md(content))+'</div></div>';document.getElementById('ca').scrollTop=document.getElementById('ca').scrollHeight}
async function send(){var sys=document.getElementById('sys')?document.getElementById('sys').value:'';var inp=document.getElementById('inp'),text=inp.value.trim();if(!text||gen)return;inp.value='';inp.style.height='auto';
document.getElementById('wel').style.display='none';document.getElementById('ca').style.display='flex';
rMsg('user',text);H.push({role:'user',content:text});chats[ci].m.push({r:'user',t:text});
if(chats[ci].t==='Nouveau chat'){chats[ci].t=text.slice(0,35);rSb()}
gen=true;document.getElementById('ld').classList.add('on');document.getElementById('ldt').textContent=agent?'Agent autonome...':'Réflexion...';document.getElementById('sb').disabled=true;
var sys=document.getElementById('sys').value,temp=parseFloat(document.getElementById('tv').textContent),maxT=parseInt(document.getElementById('mtv').textContent),t0=performance.now();
try{var r=await fetch(API,{method:'POST',headers:{'Content-Type':'application/json'},body:JSON.stringify({message:text,system:sys})});
var lat=performance.now()-t0,ls=lat<1e3?Math.round(lat)+'ms':(lat/1e3).toFixed(1)+'s',prov=r.headers.get('X-Provider')||'cerebras';
var txt='',prov='cerebras',mdl='',lat2='';
// SSE streaming reader
var reader=r.body.getReader(),dec=new TextDecoder(),buf='';
var aiDiv=document.createElement('div');aiDiv.className='msg ai';aiDiv.innerHTML='<div class="mc" id="stream-out"></div>';msgs.appendChild(aiDiv);var mc=aiDiv.querySelector('.mc');
while(true){var rd=await reader.read();if(rd.done)break;buf+=dec.decode(rd.value,{stream:true});var lns=buf.split('\n');buf=lns.pop();
for(var ln of lns){if(!ln.startsWith('data: '))continue;try{var ev=JSON.parse(ln.slice(6));if(ev.type==='token'){txt+=ev.content;mc.innerHTML=txt.replace(/\n/g,'<br>');msgs.scrollTop=msgs.scrollHeight}if(ev.type==='start'){prov=ev.provider||'cerebras';mdl=ev.model||''}if(ev.type==='done'){lat2=ev.latency_ms?ev.latency_ms+'ms':''}}catch(e){}}}
document.getElementById('ld').classList.remove('on');
mc.innerHTML+='<div style="margin-top:8px;font-size:10px;color:#999">via '+prov+' '+(mdl?mdl.split('-').slice(0,3).join('-'):'')+' '+lat2+'</div>';
gen=false;document.getElementById('sb').disabled=false;document.getElementById('inp').focus();H.push({role:'assistant',content:txt});chats[ci].m.push({r:'assistant',t:txt,o:{provider:prov}});return;var data={response:txt,provider:prov}
document.getElementById('ld').classList.remove('on');
if(data.error){rMsg('assistant','Erreur: '+(data.error.message||JSON.stringify(data.error)),{provider:prov,latency:ls})}
else{var txt=(data.content||[]).map(function(b){return b.text||''}).join('\n\n');if(!txt)txt='[Pas de réponse]';
var steps=data.agent?data.agent.steps:[];rMsg('assistant',txt,{provider:prov,latency:ls,steps:steps});
H.push({role:'assistant',content:txt});chats[ci].m.push({r:'assistant',t:txt,o:{provider:prov,latency:ls,steps:steps}});
ttok+=(data.usage?data.usage.input_tokens:0)+(data.usage?data.usage.output_tokens:0);nreq++;
document.getElementById('tc').textContent=ttok+' tok';document.getElementById('hp').textContent=prov}}
catch(e){document.getElementById('ld').classList.remove('on');rMsg('assistant','Erreur: '+e.message,{})}
gen=false;document.getElementById('sb').disabled=false;inp.focus()}
async function health(){try{var r=await fetch('/api/wevia-master-api.php?health',{signal:AbortSignal.timeout(5e3)});var d=await r.json();document.getElementById('hd').style.background=d.status==='ok'?'var(--green)':'var(--red)';document.getElementById('hp').textContent=d.primary||'cerebras-235B'}catch(e){document.getElementById('hd').style.background='var(--red)'}}
health();setInterval(health,30000);rSb();document.getElementById('inp').focus();
</script><script src="/api/live-stats.js"></script><script src="/api/live-stats.js"></script><!-- CARTO_BANNER_V1 -->
<div style="position:fixed;bottom:20px;right:20px;z-index:9999;background:linear-gradient(135deg,#141931,#2d1b5e);border:1px solid #64ffda;border-radius:12px;padding:12px 18px;box-shadow:0 4px 20px rgba(100,255,218,.3);font-family:-apple-system,Segoe UI,sans-serif;font-size:13px">
<a href="/cartographie-screens.html" style="color:#64ffda;text-decoration:none;font-weight:600;display:flex;align-items:center;gap:8px" title="Cartographie exhaustive de tous les ecrans live">
<span style="font-size:18px">&#128506;</span> Cartographie live
<span id="carto-banner-count" style="color:#8892b0;font-size:11px">3914 ecrans</span>
</a>
</div>
<script>
(function(){
fetch('/api/screens-health.php?_='+Date.now(),{cache:'no-store'}).then(r=>r.json()).then(d=>{
const c=d.counts||{}; const up=c.UP||0; const slow=c.SLOW||0; const br=c.BROKEN||0;
const el=document.getElementById('carto-banner-count');
if(el) el.innerHTML=`<span style="color:#22c55e">${up} UP</span> / <span style="color:#f59e0b">${slow} Lent</span> / <span style="color:#ef4444">${br} 5xx</span>`;
}).catch(()=>{});
})();
</script>
<!-- /CARTO_BANNER_V1 -->
</body></html>