111 lines
19 KiB
Plaintext
111 lines
19 KiB
Plaintext
<!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{font-family:'Inter',-apple-system,sans-serif;height:100vh;display:flex;color:var(--text);background:var(--bg)}
|
|
.sb{width:260px;background:var(--sbg);display:flex;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:260px;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="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>
|
|
<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,'&').replace(/</g,'<').replace(/>/g,'>')}
|
|
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>';
|
|
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></body></html>
|