Files
wevia-brain/s89-arsenal-screens/supply-chain.html
2026-04-12 23:01:36 +02:00

215 lines
14 KiB
HTML
Executable File

<?php include_once("/opt/wevads-arsenal/public/api/wevads-metrics.php"); ?>
<!DOCTYPE html><html lang="fr"><head>
<meta http-equiv="Cache-Control" content="no-cache, no-store, must-revalidate"><meta charset="UTF-8"><meta name="viewport" content="width=device-width,initial-scale=1.0">
<title>WEVADS - Supply Chain Factory — WEVADS</title>
<link href="https://fonts.googleapis.com/css2?family=DM+Sans:wght@300;400;500;600;700&family=JetBrains+Mono:wght@400;500;600&display=swap" rel="stylesheet">
<style>
*{margin:0;padding:0;box-sizing:border-box}
:root{--bg:#060a14;--s:#0c1220;--c:#111827;--b:#1e293b;--ac:#f59e0b;--gn:#060a14;--rd:#ef4444;--or:#f59e0b;--cy:#22d3ee;--pu:#a78bfa;--t:#e2e8f0;--d:#64748b;--f:'DM Sans',sans-serif;--m:'JetBrains Mono',monospace}
body{font-family:var(--f);background:#060a14;color:var(--t);min-height:100vh}
.app{max-width:1440px;margin:0 auto;padding:24px}
.hdr{display:flex;align-items:center;justify-content:space-between;margin-bottom:24px;padding-bottom:16px;border-bottom:1px solid var(--b)}
.hdr h1{font-size:22px;font-weight:700}.hdr h1 span{color:var(--ac)}
.tabs{display:flex;gap:4px;margin-bottom:20px}
.tab{padding:8px 16px;border-radius:8px;font-size:12px;font-weight:600;cursor:pointer;background:var(--c);border:1px solid var(--b);color:var(--d);transition:.2s}
.tab.active{background:var(--ac);color:#e2e8f0;border-color:var(--ac)}
.tab:hover:not(.active){border-color:var(--ac);color:var(--t)}
.panel{display:none}.panel.active{display:block}
.btn{padding:8px 18px;border:none;border-radius:10px;font-size:12px;font-weight:600;cursor:pointer;color:#fff;transition:.2s}
.btn-p{background:linear-gradient(135deg,var(--ac),#d97706)}.btn-s{background:var(--c);border:1px solid var(--b);color:var(--t)}
.btn:hover{transform:translateY(-1px)}
.g{display:grid;gap:14px;margin-bottom:20px}
.g4{grid-template-columns:repeat(4,1fr)}.g3{grid-template-columns:repeat(3,1fr)}.g2{grid-template-columns:1fr 1fr}
.kpi{background:var(--s);border:1px solid var(--b);border-radius:14px;padding:18px;text-align:center}
.kpi .val{font-family:var(--m);font-size:26px;font-weight:700}.kpi .lbl{font-size:10px;color:var(--d);text-transform:uppercase;margin-top:4px}
.kpi.green .val{color:var(--gn)}.kpi.blue .val{color:var(--cy)}.kpi.amber .val{color:var(--or)}.kpi.red .val{color:var(--rd)}
.cd{background:var(--s);border:1px solid var(--b);border-radius:14px;padding:20px}
.cd-h{display:flex;align-items:center;justify-content:space-between;margin-bottom:14px}
.cd-h h3{font-size:14px;font-weight:600}.cd-h .ct{font-size:11px;color:var(--d)}
table{width:100%;border-collapse:collapse;font-size:12px}
th{text-align:left;padding:10px 12px;color:var(--d);font-size:10px;text-transform:uppercase;border-bottom:1px solid var(--b)}
td{padding:10px 12px;border-bottom:1px solid rgba(30,41,59,.3)}
.b{padding:2px 8px;border-radius:6px;font-size:10px;font-weight:600;display:inline-block}
.b-ok{background:rgba(16,185,129,.12);color:var(--gn)}.b-w{background:rgba(245,158,11,.12);color:var(--or)}.b-err{background:rgba(239,68,68,.12);color:var(--rd)}.b-info{background:rgba(34,211,238,.12);color:var(--cy)}
.bar{height:6px;border-radius:3px;background:var(--c);overflow:hidden;flex:1}.bar-fill{height:100%;border-radius:3px;transition:.5s}
.empty{text-align:center;padding:40px;color:var(--d);font-size:13px}
.toast{position:fixed;bottom:20px;right:20px;padding:12px 20px;border-radius:10px;color:#fff;font-size:13px;z-index:9999;animation:fadeIn .3s}
.toast.info{background:var(--cy)}.toast.success{background:var(--gn)}.toast.error{background:var(--rd)}
@keyframes fadeIn{from{opacity:0;transform:translateY(10px)}to{opacity:1;transform:none}}
@media(max-width:900px){.g4,.g3{grid-template-columns:1fr 1fr}}
.wv-status{position:fixed;top:12px;right:140px;z-index:9998;background:rgba(52,211,153,.15);border:1px solid #34d399;border-radius:12px;padding:3px 10px;color:#34d399;font-size:10px;font-weight:700;font-family:'JetBrains Mono',monospace}
.sc,.card,[class*="stat-card"]{transition:all .25s ease;position:relative;overflow:hidden}
.sc:hover,.card:hover,[class*="stat-card"]:hover{transform:translateY(-2px);box-shadow:0 8px 24px rgba(0,0,0,.25)}
.sc::after,.card::after{content:'';position:absolute;bottom:0;left:0;right:0;height:2px;background:var(--cy,#22d3ee);opacity:0;transition:opacity .25s}
.sc:hover::after,.card:hover::after{opacity:.7}
.btn,.button,[class*="btn-"]{transition:all .2s ease}
.btn:hover,.button:hover{transform:translateY(-1px)}
@keyframes fadeIn{from{opacity:0;transform:translateY(8px)}to{opacity:1;transform:translateY(0)}}
.sc,.card{animation:fadeIn .4s ease both}
.sc:nth-child(2),.card:nth-child(2){animation-delay:.05s}
.sc:nth-child(3),.card:nth-child(3){animation-delay:.1s}
.sc:nth-child(4),.card:nth-child(4){animation-delay:.15s}
.sc:nth-child(5),.card:nth-child(5){animation-delay:.2s}
.sc:nth-child(6),.card:nth-child(6){animation-delay:.25s}
</style><link rel="stylesheet" href="wevads-global.css?v1770777318">
</head><body>
<div class="app">
<div class="hdr"><h1>🏭 <span>Supply Chain</span> Factory</h1>
<p style="font-size:12px;color:#64748b;margin:6px 0 16px;max-width:600px;line-height:1.6">Supply Chain — chaîne complète : domaines → IPs → comptes → envoi.</p><div style="display:flex;gap:8px"><button class="btn btn-s" onclick="loadAll()">🔄 Refresh</button><button class="btn btn-p" onclick="runAudit()">🔍 Run Audit</button></div></div>
<div class="tabs">
<div class="tab active" data-tab="overview">📊 Overview</div>
<div class="tab" data-tab="domains">🌐 Domains</div>
<div class="tab" data-tab="servers">🖥️ Servers</div>
<div class="tab" data-tab="providers">☁️ Providers</div>
<div class="tab" data-tab="costs">💰 Costs</div>
</div>
<div class="panel active" id="panel-overview">
<div class="g g4" id="kpis"></div>
<div class="g g2">
<div class="cd"><div class="cd-h"><h3>📡 Infrastructure Status</h3></div><div id="infra-status"></div></div>
<div class="cd"><div class="cd-h"><h3>⚠️ Alerts</h3><span class="ct" id="alert-ct"></span></div><div id="alerts"></div></div>
</div>
</div>
<div class="panel" id="panel-domains">
<div class="cd"><div class="cd-h"><h3>🌐 Domain Inventory</h3><span class="ct" id="dom-ct"></span></div><div id="dom-table"></div></div>
</div>
<div class="panel" id="panel-servers">
<div class="cd"><div class="cd-h"><h3>🖥️ Server Fleet</h3></div><div id="srv-table"></div></div>
</div>
<div class="panel" id="panel-providers">
<div class="cd"><div class="cd-h"><h3>☁️ Cloud Providers</h3></div><div id="prov-table"></div></div>
</div>
<div class="panel" id="panel-costs">
<div class="g g3" id="cost-kpis"></div>
<div class="cd"><div class="cd-h"><h3>💰 Cost Breakdown</h3></div><div id="cost-table"></div></div>
</div>
</div>
<script>
const API='/api/';
function toast(m,t='info'){const d=document.createElement('div');d.className='toast '+t;d.textContent=m;document.body.appendChild(d);setTimeout(()=>d.remove(),4000)}
function esc(s){if(!s)return'';const d=document.createElement('div');d.textContent=s;return d.innerHTML}
document.querySelectorAll('.tab').forEach(t=>{t.addEventListener('click',()=>{
document.querySelectorAll('.tab').forEach(x=>x.classList.remove('active'));
document.querySelectorAll('.panel').forEach(x=>x.classList.remove('active'));
t.classList.add('active');document.getElementById('panel-'+t.dataset.tab).classList.add('active');
})});
const infra={
servers:[
{name:'Hetzner Main',ip:'157.180.25.208',role:'WEVADS Core',status:'online',cpu:45,ram:62,disk:78},
{name:'OVH Tracking',ip:'151.80.235.110',role:'Track Pipeline',status:'online',cpu:22,ram:38,disk:34},
{name:'Consulting',ip:'46.62.220.135',role:'Website',status:'online',cpu:12,ram:28,disk:45}
],
domains:[
{name:'culturellemejean.charity',type:'Tracking',dns:'Cloudflare',ssl:'Active',status:'ok'},
{name:'babosur.com',type:'CX3 Tracking',dns:'Cloudflare',ssl:'Active',status:'ok'}
],
providers:[
{name:'Hetzner',type:'Dedicated',services:'Compute, Storage',monthly:45,status:'ok'},
{name:'OVH',type:'VPS',services:'Tracking',monthly:12,status:'ok'},
{name:'Cloudflare',type:'CDN/DNS',services:'DNS, WAF, Workers',monthly:0,status:'ok'},
{name:'Huawei Cloud',type:'Cloud',services:'Object Storage',monthly:8,status:'ok'},
{name:'AWS S3',type:'Storage',services:'Backup',monthly:5,status:'ok'}
]
};
function loadAll(){
// KPIs
document.getElementById('kpis').innerHTML=`
<div class="kpi green"><div class="val">${infra.servers.length}</div><div class="lbl">Servers</div></div>
<div class="kpi blue"><div class="val">${infra.domains.length}</div><div class="lbl">Domains</div></div>
<div class="kpi amber"><div class="val">${infra.providers.length}</div><div class="lbl">Providers</div></div>
<div class="kpi green"><div class="val">€${infra.providers.reduce((s,p)=>s+p.monthly,0)}/mo</div><div class="lbl">Total Cost</div></div>`;
// Infra status
document.getElementById('infra-status').innerHTML=infra.servers.map(s=>`
<div style="display:flex;align-items:center;gap:12px;padding:10px;background:var(--c);border:1px solid var(--b);border-radius:10px;margin-bottom:8px">
<span style="color:var(--gn)">●</span>
<div style="flex:1"><div style="font-weight:600;font-size:13px">${s.name}</div><div style="font-size:11px;color:var(--d)">${s.ip}${s.role}</div></div>
<div style="display:flex;gap:16px;font-family:var(--m);font-size:11px">
<span title="CPU">🔲 ${s.cpu}%</span><span title="RAM">💾 ${s.ram}%</span><span title="Disk">💿 ${s.disk}%</span>
</div>
</div>`).join('');
// Alerts
const alerts=[];
infra.servers.forEach(s=>{if(s.disk>75)alerts.push({level:'warn',msg:`${s.name} disk at ${s.disk}%`})});
if(alerts.length===0)alerts.push({level:'ok',msg:'No active alerts — all systems healthy'});
document.getElementById('alert-ct').textContent=alerts.length;
document.getElementById('alerts').innerHTML=alerts.map(a=>`
<div style="display:flex;align-items:center;gap:8px;padding:8px;margin-bottom:4px;font-size:12px">
<span style="color:${a.level==='ok'?'var(--gn)':'var(--or)'}">●</span>${a.msg}
</div>`).join('');
// Domains
document.getElementById('dom-ct').textContent=infra.domains.length;
document.getElementById('dom-table').innerHTML=`<table><tr><th>Domain</th><th>Type</th><th>DNS</th><th>SSL</th><th>Status</th></tr>
${infra.domains.map(d=>`<tr><td style="font-family:var(--m);font-size:11px">${d.name}</td><td>${d.type}</td><td>${d.dns}</td><td>${d.ssl}</td><td><span class="b b-ok">${d.status}</span></td></tr>`).join('')}</table>`;
// Servers
document.getElementById('srv-table').innerHTML=`<table><tr><th>Name</th><th>IP</th><th>Role</th><th>CPU</th><th>RAM</th><th>Disk</th><th>Status</th></tr>
${infra.servers.map(s=>`<tr><td style="font-weight:600">${s.name}</td><td style="font-family:var(--m);font-size:11px">${s.ip}</td><td>${s.role}</td>
<td><div style="display:flex;align-items:center;gap:6px"><div class="bar"><div class="bar-fill" style="width:${s.cpu}%;background:${s.cpu>80?'var(--rd)':'var(--gn)'}"></div></div><span style="font-size:10px">${s.cpu}%</span></div></td>
<td><div style="display:flex;align-items:center;gap:6px"><div class="bar"><div class="bar-fill" style="width:${s.ram}%;background:${s.ram>80?'var(--rd)':'var(--cy)'}"></div></div><span style="font-size:10px">${s.ram}%</span></div></td>
<td><div style="display:flex;align-items:center;gap:6px"><div class="bar"><div class="bar-fill" style="width:${s.disk}%;background:${s.disk>80?'var(--rd)':'var(--or)'}"></div></div><span style="font-size:10px">${s.disk}%</span></div></td>
<td><span class="b b-ok">Online</span></td></tr>`).join('')}</table>`;
// Providers
document.getElementById('prov-table').innerHTML=`<table><tr><th>Provider</th><th>Type</th><th>Services</th><th>Monthly</th><th>Status</th></tr>
${infra.providers.map(p=>`<tr><td style="font-weight:600">${p.name}</td><td><span class="b b-info">${p.type}</span></td><td>${p.services}</td><td style="font-family:var(--m)">€${p.monthly}</td><td><span class="b b-ok">Active</span></td></tr>`).join('')}</table>`;
// Costs
const total=infra.providers.reduce((s,p)=>s+p.monthly,0);
document.getElementById('cost-kpis').innerHTML=`
<div class="kpi amber"><div class="val">€${total}</div><div class="lbl">Monthly Total</div></div>
<div class="kpi blue"><div class="val">€${(total*12).toFixed(0)}</div><div class="lbl">Annual Estimate</div></div>
<div class="kpi green"><div class="val">€${(total/3).toFixed(0)}</div><div class="lbl">Per Server</div></div>`;
document.getElementById('cost-table').innerHTML=`<table><tr><th>Provider</th><th>Service</th><th>Monthly</th><th>% of Total</th></tr>
${infra.providers.map(p=>`<tr><td>${p.name}</td><td>${p.services}</td><td style="font-family:var(--m)">€${p.monthly}</td>
<td><div style="display:flex;align-items:center;gap:8px"><div class="bar"><div class="bar-fill" style="width:${(p.monthly/total*100).toFixed(0)}%;background:var(--ac)"></div></div><span style="font-size:10px">${(p.monthly/total*100).toFixed(0)}%</span></div></td></tr>`).join('')}</table>`;
}
function runAudit(){toast('Running supply chain audit...','info');setTimeout(()=>{toast('Audit complete — all systems healthy','success')},2000)}
loadAll();
</script><script>
var _API="/api/supply-chain-factory.php";
async function _autoLoad(){
try{
var r=await fetch(_API+"?action=status");
var d=await r.json();
if(!d)return;
var src=d.data||d;
if(typeof src==="object"&&!Array.isArray(src)){
var entries=Object.entries(src);
var statEls=document.querySelectorAll(".st .n,.kpi-value,.stat-value,[id^=k],[id^=s-]");
statEls.forEach(function(el,idx){
if(idx<entries.length){
var v=entries[idx][1];
if(typeof v==="number")v=v.toLocaleString();
if(typeof v==="string"||typeof v==="number")el.textContent=v;
}
});
}
var si=document.querySelector(".status-dot,.live-dot,[class*=live]");
if(si)si.style.color="#22c55e";
}catch(e){console.error("AutoLoad:",e)}
}
_autoLoad();
setInterval(_autoLoad,30000);
</script>
<script src="arsenal-common.js?v1770778169">
<?php include("/opt/wevads-arsenal/public/universal-drill.html"); ?>
</body></html>
</script>